From 821cddefe814c45fd0f7f0fb2ef6f7aba4ad3823 Mon Sep 17 00:00:00 2001 From: HaHaWTH <102713261+HaHaWTH@users.noreply.github.com> Date: Fri, 1 Nov 2024 20:26:04 +0800 Subject: [PATCH 1/7] Working on fix Async Pathfinding (Stage 1) --- .../server/0040-Petal-Async-Pathfinding.patch | 76 ++++++++++++------- 1 file changed, 48 insertions(+), 28 deletions(-) diff --git a/patches/server/0040-Petal-Async-Pathfinding.patch b/patches/server/0040-Petal-Async-Pathfinding.patch index 4ee0d7b91..23320aa54 100644 --- a/patches/server/0040-Petal-Async-Pathfinding.patch +++ b/patches/server/0040-Petal-Async-Pathfinding.patch @@ -27,7 +27,7 @@ index 63b827d91a935d6b6f04266eea682da97af79cf2..02d7180e5b932dd8c7e8867f1334cbc4 } diff --git a/src/main/java/net/minecraft/world/entity/ai/behavior/AcquirePoi.java b/src/main/java/net/minecraft/world/entity/ai/behavior/AcquirePoi.java -index 4d2b6e69ed98aca98ffc50fefc6e535c1afb759d..fe7e9e194634f6bfff2aab6e496a35fb85554e5a 100644 +index 4d2b6e69ed98aca98ffc50fefc6e535c1afb759d..299a26d731da4b56887b427cf3551d84f0e231e6 100644 --- a/src/main/java/net/minecraft/world/entity/ai/behavior/AcquirePoi.java +++ b/src/main/java/net/minecraft/world/entity/ai/behavior/AcquirePoi.java @@ -85,6 +85,38 @@ public class AcquirePoi { @@ -40,7 +40,7 @@ index 4d2b6e69ed98aca98ffc50fefc6e535c1afb759d..fe7e9e194634f6bfff2aab6e496a35fb + Path possiblePath = findPathToPois(entity, set); + + // wait on the path to be processed -+ org.dreeam.leaf.async.path.AsyncPathProcessor.awaitProcessing(entity, possiblePath, path -> { ++ org.dreeam.leaf.async.path.AsyncPathProcessor.awaitProcessing(possiblePath, path -> { + // read canReach check + if (path == null || !path.canReach()) { + for (Pair, BlockPos> pair : set) { @@ -203,7 +203,7 @@ index 2a7a26ca447cc78f24e61a2bf557411c31eb16b2..4010cb7ad8897995c8b850f9279aad2a private boolean tryComputePath(Mob entity, WalkTarget walkTarget, long time) { BlockPos blockPos = walkTarget.getTarget().currentBlockPosition(); diff --git a/src/main/java/net/minecraft/world/entity/ai/behavior/SetClosestHomeAsWalkTarget.java b/src/main/java/net/minecraft/world/entity/ai/behavior/SetClosestHomeAsWalkTarget.java -index 6802e0c4d331c7125114dd86409f6a110465ab82..84a36ef3e98af24a24a25b83826f44bb7f746bcc 100644 +index 6802e0c4d331c7125114dd86409f6a110465ab82..601f43615cb55142125e21411f651318ee760e9f 100644 --- a/src/main/java/net/minecraft/world/entity/ai/behavior/SetClosestHomeAsWalkTarget.java +++ b/src/main/java/net/minecraft/world/entity/ai/behavior/SetClosestHomeAsWalkTarget.java @@ -60,6 +60,26 @@ public class SetClosestHomeAsWalkTarget { @@ -216,7 +216,7 @@ index 6802e0c4d331c7125114dd86409f6a110465ab82..84a36ef3e98af24a24a25b83826f44bb + Path possiblePath = AcquirePoi.findPathToPois(entity, set); + + // wait on the path to be processed -+ org.dreeam.leaf.async.path.AsyncPathProcessor.awaitProcessing(entity, possiblePath, path -> { ++ org.dreeam.leaf.async.path.AsyncPathProcessor.awaitProcessing(possiblePath, path -> { + if (path == null || !path.canReach() || mutableInt.getValue() < 5) { // read canReach check + long2LongMap.long2LongEntrySet().removeIf(entry -> entry.getLongValue() < mutableLong.getValue()); + return; @@ -356,7 +356,7 @@ index 62634bedd97c5be9ecce24ab0cff205715a68da8..5266cee05a00fefba98a10eb91bb477f } diff --git a/src/main/java/net/minecraft/world/entity/ai/navigation/PathNavigation.java b/src/main/java/net/minecraft/world/entity/ai/navigation/PathNavigation.java -index aea01c46bf59ed811356180436fc0789e354d981..8ba2c3855b69ad9ca717675e70df1055cb4a677a 100644 +index aea01c46bf59ed811356180436fc0789e354d981..542c63c4304d97772988dea13edbcd31c14cb955 100644 --- a/src/main/java/net/minecraft/world/entity/ai/navigation/PathNavigation.java +++ b/src/main/java/net/minecraft/world/entity/ai/navigation/PathNavigation.java @@ -152,6 +152,10 @@ public abstract class PathNavigation { @@ -379,7 +379,7 @@ index aea01c46bf59ed811356180436fc0789e354d981..8ba2c3855b69ad9ca717675e70df1055 + // assign early a target position. most calls will only have 1 position + if (!positions.isEmpty()) this.targetPos = positions.iterator().next(); + -+ org.dreeam.leaf.async.path.AsyncPathProcessor.awaitProcessing(mob, path, processedPath -> { ++ org.dreeam.leaf.async.path.AsyncPathProcessor.awaitProcessing(path, processedPath -> { + // check that processing didn't take so long that we calculated a new path + if (processedPath != this.path) return; + @@ -468,7 +468,7 @@ index ce7398a617abe6e800c1e014b3ac5c970eb15c8a..dbf43209417d8453ff39839392eba45b } diff --git a/src/main/java/net/minecraft/world/entity/ai/sensing/NearestBedSensor.java b/src/main/java/net/minecraft/world/entity/ai/sensing/NearestBedSensor.java -index 9104d7010bda6f9f73b478c11490ef9c53f76da2..7abf4ecd1153be597efaa12ec330d20e4d49039e 100644 +index 9104d7010bda6f9f73b478c11490ef9c53f76da2..fb12b8581ebaccc12dc336cc73a847d75b06c421 100644 --- a/src/main/java/net/minecraft/world/entity/ai/sensing/NearestBedSensor.java +++ b/src/main/java/net/minecraft/world/entity/ai/sensing/NearestBedSensor.java @@ -57,6 +57,26 @@ public class NearestBedSensor extends Sensor { @@ -478,7 +478,7 @@ index 9104d7010bda6f9f73b478c11490ef9c53f76da2..7abf4ecd1153be597efaa12ec330d20e + // Kaiiju start - await on async path processing + if (org.dreeam.leaf.config.modules.async.AsyncPathfinding.enabled) { + Path possiblePath = AcquirePoi.findPathToPois(entity, new java.util.HashSet<>(poiposes)); -+ org.dreeam.leaf.async.path.AsyncPathProcessor.awaitProcessing(entity, possiblePath, path -> { ++ org.dreeam.leaf.async.path.AsyncPathProcessor.awaitProcessing(possiblePath, path -> { + // read canReach check + if ((path == null || !path.canReach()) && this.triedCount < 5) { + this.batchCache.long2LongEntrySet().removeIf(entry -> entry.getLongValue() < this.lastUpdate); @@ -802,10 +802,10 @@ index 6308822f819d7cb84c8070c8a7eec1a3f822114b..e49851557f991ca1fc2f78abfb819609 public SwimNodeEvaluator(boolean canJumpOutOfWater) { diff --git a/src/main/java/org/dreeam/leaf/async/path/AsyncPath.java b/src/main/java/org/dreeam/leaf/async/path/AsyncPath.java new file mode 100644 -index 0000000000000000000000000000000000000000..2f5aff1f0e2aca0a8bfeab27c4ab027ffc21055d +index 0000000000000000000000000000000000000000..ba44944fd043e3982477bfee2c48a0e765d62db0 --- /dev/null +++ b/src/main/java/org/dreeam/leaf/async/path/AsyncPath.java -@@ -0,0 +1,287 @@ +@@ -0,0 +1,295 @@ +package org.dreeam.leaf.async.path; + +import net.minecraft.core.BlockPos; @@ -829,7 +829,7 @@ index 0000000000000000000000000000000000000000..2f5aff1f0e2aca0a8bfeab27c4ab027f + /** + * marks whether this async path has been processed + */ -+ private volatile boolean processed = false; ++ private volatile PathProcessState processState = PathProcessState.WAITING; + + /** + * runnables waiting for this to be processed @@ -886,14 +886,14 @@ index 0000000000000000000000000000000000000000..2f5aff1f0e2aca0a8bfeab27c4ab027f + + @Override + public boolean isProcessed() { -+ return this.processed; ++ return this.processState == PathProcessState.COMPLETED; + } + + /** + * returns the future representing the processing state of this path + */ + public synchronized void postProcessing(@NotNull Runnable runnable) { -+ if (this.processed) { ++ if (isProcessed()) { + runnable.run(); + } else { + this.postProcessing.add(runnable); @@ -918,10 +918,13 @@ index 0000000000000000000000000000000000000000..2f5aff1f0e2aca0a8bfeab27c4ab027f + * starts processing this path + */ + public synchronized void process() { -+ if (this.processed) { ++ if (this.processState == PathProcessState.COMPLETED || ++ this.processState == PathProcessState.PROCESSING) { + return; + } + ++ processState = PathProcessState.PROCESSING; ++ + final Path bestPath = this.pathSupplier.get(); + + this.nodes.addAll(bestPath.nodes); // we mutate this list to reuse the logic in Path @@ -929,18 +932,19 @@ index 0000000000000000000000000000000000000000..2f5aff1f0e2aca0a8bfeab27c4ab027f + this.distToTarget = bestPath.getDistToTarget(); + this.canReach = bestPath.canReach(); + -+ this.processed = true; ++ processState = PathProcessState.COMPLETED; + + for (Runnable runnable : this.postProcessing) { + runnable.run(); -+ } ++ } // Run tasks after processing + } + + /** + * if this path is accessed while it hasn't processed, just process it in-place + */ + private void checkProcessed() { -+ if (!this.processed) { ++ if (this.processState == PathProcessState.WAITING || ++ this.processState == PathProcessState.PROCESSING) { // Block if we are on processing + this.process(); + } + } @@ -976,7 +980,7 @@ index 0000000000000000000000000000000000000000..2f5aff1f0e2aca0a8bfeab27c4ab027f + + @Override + public boolean isDone() { -+ return this.isProcessed() && super.isDone(); ++ return this.processState == PathProcessState.COMPLETED && super.isDone(); + } + + @Override @@ -1092,19 +1096,23 @@ index 0000000000000000000000000000000000000000..2f5aff1f0e2aca0a8bfeab27c4ab027f + + return super.hasNext(); + } ++ ++ public PathProcessState getProcessState() { ++ return processState; ++ } +} diff --git a/src/main/java/org/dreeam/leaf/async/path/AsyncPathProcessor.java b/src/main/java/org/dreeam/leaf/async/path/AsyncPathProcessor.java new file mode 100644 -index 0000000000000000000000000000000000000000..3eb86fc2e0ea28be18e23dd2c060e043f1fede93 +index 0000000000000000000000000000000000000000..192edd0fdc8e2fd7fa11bef416544810c94f292b --- /dev/null +++ b/src/main/java/org/dreeam/leaf/async/path/AsyncPathProcessor.java -@@ -0,0 +1,52 @@ +@@ -0,0 +1,51 @@ +package org.dreeam.leaf.async.path; + +import com.google.common.util.concurrent.ThreadFactoryBuilder; + ++import net.minecraft.server.MinecraftServer; +import net.minecraft.world.level.pathfinder.Path; -+import net.minecraft.world.entity.Entity; + +import org.jetbrains.annotations.NotNull; +import org.jetbrains.annotations.Nullable; @@ -1137,14 +1145,13 @@ index 0000000000000000000000000000000000000000..3eb86fc2e0ea28be18e23dd2c060e043 + * the consumer will be immediately invoked if the path is already processed + * the consumer will always be called on the main thread + * -+ * @param entity affected entity + * @param path a path to wait on + * @param afterProcessing a consumer to be called + */ -+ public static void awaitProcessing(Entity entity, @Nullable Path path, Consumer<@Nullable Path> afterProcessing) { ++ public static void awaitProcessing(@Nullable Path path, Consumer<@Nullable Path> afterProcessing) { + if (path != null && !path.isProcessed() && path instanceof AsyncPath asyncPath) { + asyncPath.postProcessing(() -> -+ entity.getBukkitEntity().taskScheduler.schedule(nmsEntity -> afterProcessing.accept(path), null, 1) ++ MinecraftServer.getServer().scheduleOnMain(() -> afterProcessing.accept(path)) + ); + } else { + afterProcessing.accept(path); @@ -1153,12 +1160,13 @@ index 0000000000000000000000000000000000000000..3eb86fc2e0ea28be18e23dd2c060e043 +} diff --git a/src/main/java/org/dreeam/leaf/async/path/NodeEvaluatorCache.java b/src/main/java/org/dreeam/leaf/async/path/NodeEvaluatorCache.java new file mode 100644 -index 0000000000000000000000000000000000000000..7a28c0ef27efbab28274c007068b5c82073a1987 +index 0000000000000000000000000000000000000000..b147a9675a45bd1306e4cf2a4f155025ce1ae1bf --- /dev/null +++ b/src/main/java/org/dreeam/leaf/async/path/NodeEvaluatorCache.java @@ -0,0 +1,45 @@ +package org.dreeam.leaf.async.path; + ++import ca.spottedleaf.concurrentutil.collection.MultiThreadedQueue; +import net.minecraft.world.level.pathfinder.NodeEvaluator; + +import org.apache.commons.lang.Validate; @@ -1167,14 +1175,13 @@ index 0000000000000000000000000000000000000000..7a28c0ef27efbab28274c007068b5c82 +import java.util.Map; +import java.util.Queue; +import java.util.concurrent.ConcurrentHashMap; -+import java.util.concurrent.ConcurrentLinkedQueue; + +public class NodeEvaluatorCache { -+ private static final Map> threadLocalNodeEvaluators = new ConcurrentHashMap<>(); ++ private static final Map> threadLocalNodeEvaluators = new ConcurrentHashMap<>(); + private static final Map nodeEvaluatorToGenerator = new ConcurrentHashMap<>(); + + private static @NotNull Queue getQueueForFeatures(@NotNull NodeEvaluatorFeatures nodeEvaluatorFeatures) { -+ return threadLocalNodeEvaluators.computeIfAbsent(nodeEvaluatorFeatures, key -> new ConcurrentLinkedQueue<>()); ++ return threadLocalNodeEvaluators.computeIfAbsent(nodeEvaluatorFeatures, key -> new MultiThreadedQueue<>()); + } + + public static @NotNull NodeEvaluator takeNodeEvaluator(@NotNull NodeEvaluatorGenerator generator, @NotNull NodeEvaluator localNodeEvaluator) { @@ -1272,6 +1279,19 @@ index 0000000000000000000000000000000000000000..c0527323c42acf7e4728237e268f075e + return WALK; + } +} +diff --git a/src/main/java/org/dreeam/leaf/async/path/PathProcessState.java b/src/main/java/org/dreeam/leaf/async/path/PathProcessState.java +new file mode 100644 +index 0000000000000000000000000000000000000000..73f30b733fb93f5cfbf9e14800b055b9053f6383 +--- /dev/null ++++ b/src/main/java/org/dreeam/leaf/async/path/PathProcessState.java +@@ -0,0 +1,7 @@ ++package org.dreeam.leaf.async.path; ++ ++public enum PathProcessState { ++ WAITING, ++ PROCESSING, ++ COMPLETED, ++} diff --git a/src/main/java/org/dreeam/leaf/config/modules/async/AsyncPathfinding.java b/src/main/java/org/dreeam/leaf/config/modules/async/AsyncPathfinding.java new file mode 100644 index 0000000000000000000000000000000000000000..b45d738fb982b0a245ee5fda8c4abd0df0441f74 From e9d0f31f92a8bf4dd6549658f7f3e2c6ef565323 Mon Sep 17 00:00:00 2001 From: HaHaWTH <102713261+HaHaWTH@users.noreply.github.com> Date: Sat, 2 Nov 2024 00:22:35 +0800 Subject: [PATCH 2/7] No need to extend TickThread --- .../server/0113-Multithreaded-Tracker.patch | 25 ++----------------- 1 file changed, 2 insertions(+), 23 deletions(-) diff --git a/patches/server/0113-Multithreaded-Tracker.patch b/patches/server/0113-Multithreaded-Tracker.patch index d16685a76..63b8ee778 100644 --- a/patches/server/0113-Multithreaded-Tracker.patch +++ b/patches/server/0113-Multithreaded-Tracker.patch @@ -367,15 +367,14 @@ index 14ceb3308474e76220bd64b0254df3f2925d4206..6cd45791b19df76e367d2693bce349c6 private final net.minecraft.world.entity.LivingEntity entity; // Purpur diff --git a/src/main/java/org/dreeam/leaf/async/tracker/MultithreadedTracker.java b/src/main/java/org/dreeam/leaf/async/tracker/MultithreadedTracker.java new file mode 100644 -index 0000000000000000000000000000000000000000..587c2c5b75dedfd8e218a8e26284ef83f56a0d51 +index 0000000000000000000000000000000000000000..1fc19cc2945eff7bd5d3d3da826f4f973e5a7eb8 --- /dev/null +++ b/src/main/java/org/dreeam/leaf/async/tracker/MultithreadedTracker.java -@@ -0,0 +1,189 @@ +@@ -0,0 +1,168 @@ +package org.dreeam.leaf.async.tracker; + +import ca.spottedleaf.moonrise.common.list.ReferenceList; +import ca.spottedleaf.moonrise.common.misc.NearbyPlayers; -+import ca.spottedleaf.moonrise.common.util.TickThread; +import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemServerLevel; +import ca.spottedleaf.moonrise.patches.chunk_system.level.entity.server.ServerEntityLookup; +import ca.spottedleaf.moonrise.patches.entity_tracker.EntityTrackerEntity; @@ -392,36 +391,16 @@ index 0000000000000000000000000000000000000000..587c2c5b75dedfd8e218a8e26284ef83 +import java.util.concurrent.LinkedBlockingQueue; +import java.util.concurrent.ThreadPoolExecutor; +import java.util.concurrent.TimeUnit; -+import java.util.concurrent.atomic.AtomicInteger; + +public class MultithreadedTracker { + + private static final Logger LOGGER = LogManager.getLogger("MultithreadedTracker"); -+ public static class MultithreadedTrackerThread extends TickThread { -+ private static final AtomicInteger THREAD_COUNTER = new AtomicInteger(0); -+ public MultithreadedTrackerThread(Runnable run, String name) { -+ super(run, name, THREAD_COUNTER.incrementAndGet()); -+ } -+ -+ @Override -+ public void run() { -+ super.run(); -+ } -+ } + private static final Executor trackerExecutor = new ThreadPoolExecutor( + 1, + org.dreeam.leaf.config.modules.async.MultithreadedTracker.asyncEntityTrackerMaxThreads, + org.dreeam.leaf.config.modules.async.MultithreadedTracker.asyncEntityTrackerKeepalive, TimeUnit.SECONDS, + new LinkedBlockingQueue<>(), + new ThreadFactoryBuilder() -+ .setThreadFactory( -+ r -> new MultithreadedTrackerThread(r, "Leaf Async Tracker Thread") { -+ @Override -+ public void run() { -+ r.run(); -+ } -+ } -+ ) + .setNameFormat("Leaf Async Tracker Thread - %d") + .setPriority(Thread.NORM_PRIORITY - 2) + .build()); From ad2ea6b4585e635e28644c35b266fbff98edc6ce Mon Sep 17 00:00:00 2001 From: HaHaWTH <102713261+HaHaWTH@users.noreply.github.com> Date: Sat, 2 Nov 2024 00:29:55 +0800 Subject: [PATCH 3/7] Revert "No need to extend TickThread" This reverts commit e9d0f31f92a8bf4dd6549658f7f3e2c6ef565323. --- .../server/0113-Multithreaded-Tracker.patch | 25 +++++++++++++++++-- 1 file changed, 23 insertions(+), 2 deletions(-) diff --git a/patches/server/0113-Multithreaded-Tracker.patch b/patches/server/0113-Multithreaded-Tracker.patch index 63b8ee778..d16685a76 100644 --- a/patches/server/0113-Multithreaded-Tracker.patch +++ b/patches/server/0113-Multithreaded-Tracker.patch @@ -367,14 +367,15 @@ index 14ceb3308474e76220bd64b0254df3f2925d4206..6cd45791b19df76e367d2693bce349c6 private final net.minecraft.world.entity.LivingEntity entity; // Purpur diff --git a/src/main/java/org/dreeam/leaf/async/tracker/MultithreadedTracker.java b/src/main/java/org/dreeam/leaf/async/tracker/MultithreadedTracker.java new file mode 100644 -index 0000000000000000000000000000000000000000..1fc19cc2945eff7bd5d3d3da826f4f973e5a7eb8 +index 0000000000000000000000000000000000000000..587c2c5b75dedfd8e218a8e26284ef83f56a0d51 --- /dev/null +++ b/src/main/java/org/dreeam/leaf/async/tracker/MultithreadedTracker.java -@@ -0,0 +1,168 @@ +@@ -0,0 +1,189 @@ +package org.dreeam.leaf.async.tracker; + +import ca.spottedleaf.moonrise.common.list.ReferenceList; +import ca.spottedleaf.moonrise.common.misc.NearbyPlayers; ++import ca.spottedleaf.moonrise.common.util.TickThread; +import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemServerLevel; +import ca.spottedleaf.moonrise.patches.chunk_system.level.entity.server.ServerEntityLookup; +import ca.spottedleaf.moonrise.patches.entity_tracker.EntityTrackerEntity; @@ -391,16 +392,36 @@ index 0000000000000000000000000000000000000000..1fc19cc2945eff7bd5d3d3da826f4f97 +import java.util.concurrent.LinkedBlockingQueue; +import java.util.concurrent.ThreadPoolExecutor; +import java.util.concurrent.TimeUnit; ++import java.util.concurrent.atomic.AtomicInteger; + +public class MultithreadedTracker { + + private static final Logger LOGGER = LogManager.getLogger("MultithreadedTracker"); ++ public static class MultithreadedTrackerThread extends TickThread { ++ private static final AtomicInteger THREAD_COUNTER = new AtomicInteger(0); ++ public MultithreadedTrackerThread(Runnable run, String name) { ++ super(run, name, THREAD_COUNTER.incrementAndGet()); ++ } ++ ++ @Override ++ public void run() { ++ super.run(); ++ } ++ } + private static final Executor trackerExecutor = new ThreadPoolExecutor( + 1, + org.dreeam.leaf.config.modules.async.MultithreadedTracker.asyncEntityTrackerMaxThreads, + org.dreeam.leaf.config.modules.async.MultithreadedTracker.asyncEntityTrackerKeepalive, TimeUnit.SECONDS, + new LinkedBlockingQueue<>(), + new ThreadFactoryBuilder() ++ .setThreadFactory( ++ r -> new MultithreadedTrackerThread(r, "Leaf Async Tracker Thread") { ++ @Override ++ public void run() { ++ r.run(); ++ } ++ } ++ ) + .setNameFormat("Leaf Async Tracker Thread - %d") + .setPriority(Thread.NORM_PRIORITY - 2) + .build()); From 96ff2c8f74f277ba8e1006173328552ed2ac1115 Mon Sep 17 00:00:00 2001 From: HaHaWTH <102713261+HaHaWTH@users.noreply.github.com> Date: Sat, 2 Nov 2024 01:44:02 +0800 Subject: [PATCH 4/7] No need to extend TickThread --- patches/server/0113-Multithreaded-Tracker.patch | 15 ++++----------- 1 file changed, 4 insertions(+), 11 deletions(-) diff --git a/patches/server/0113-Multithreaded-Tracker.patch b/patches/server/0113-Multithreaded-Tracker.patch index d16685a76..4372ce6bc 100644 --- a/patches/server/0113-Multithreaded-Tracker.patch +++ b/patches/server/0113-Multithreaded-Tracker.patch @@ -367,15 +367,14 @@ index 14ceb3308474e76220bd64b0254df3f2925d4206..6cd45791b19df76e367d2693bce349c6 private final net.minecraft.world.entity.LivingEntity entity; // Purpur diff --git a/src/main/java/org/dreeam/leaf/async/tracker/MultithreadedTracker.java b/src/main/java/org/dreeam/leaf/async/tracker/MultithreadedTracker.java new file mode 100644 -index 0000000000000000000000000000000000000000..587c2c5b75dedfd8e218a8e26284ef83f56a0d51 +index 0000000000000000000000000000000000000000..1ecaa9f4b4843c05af2139ae3e4b0dd00a78a917 --- /dev/null +++ b/src/main/java/org/dreeam/leaf/async/tracker/MultithreadedTracker.java -@@ -0,0 +1,189 @@ +@@ -0,0 +1,182 @@ +package org.dreeam.leaf.async.tracker; + +import ca.spottedleaf.moonrise.common.list.ReferenceList; +import ca.spottedleaf.moonrise.common.misc.NearbyPlayers; -+import ca.spottedleaf.moonrise.common.util.TickThread; +import ca.spottedleaf.moonrise.patches.chunk_system.level.ChunkSystemServerLevel; +import ca.spottedleaf.moonrise.patches.chunk_system.level.entity.server.ServerEntityLookup; +import ca.spottedleaf.moonrise.patches.entity_tracker.EntityTrackerEntity; @@ -392,17 +391,11 @@ index 0000000000000000000000000000000000000000..587c2c5b75dedfd8e218a8e26284ef83 +import java.util.concurrent.LinkedBlockingQueue; +import java.util.concurrent.ThreadPoolExecutor; +import java.util.concurrent.TimeUnit; -+import java.util.concurrent.atomic.AtomicInteger; + +public class MultithreadedTracker { + + private static final Logger LOGGER = LogManager.getLogger("MultithreadedTracker"); -+ public static class MultithreadedTrackerThread extends TickThread { -+ private static final AtomicInteger THREAD_COUNTER = new AtomicInteger(0); -+ public MultithreadedTrackerThread(Runnable run, String name) { -+ super(run, name, THREAD_COUNTER.incrementAndGet()); -+ } -+ ++ public static class MultithreadedTrackerThread extends Thread { + @Override + public void run() { + super.run(); @@ -415,7 +408,7 @@ index 0000000000000000000000000000000000000000..587c2c5b75dedfd8e218a8e26284ef83 + new LinkedBlockingQueue<>(), + new ThreadFactoryBuilder() + .setThreadFactory( -+ r -> new MultithreadedTrackerThread(r, "Leaf Async Tracker Thread") { ++ r -> new MultithreadedTrackerThread() { + @Override + public void run() { + r.run(); From 98973726b3e03f6b1ac8d4b1769fdb0bc80dd6bd Mon Sep 17 00:00:00 2001 From: HaHaWTH <102713261+HaHaWTH@users.noreply.github.com> Date: Sat, 2 Nov 2024 15:04:08 +0800 Subject: [PATCH 5/7] TT20 lag compensation --- .../server/0125-TT20-Lag-compensation.patch | 216 ++++++++++++++++++ 1 file changed, 216 insertions(+) create mode 100644 patches/server/0125-TT20-Lag-compensation.patch diff --git a/patches/server/0125-TT20-Lag-compensation.patch b/patches/server/0125-TT20-Lag-compensation.patch new file mode 100644 index 000000000..61ea14003 --- /dev/null +++ b/patches/server/0125-TT20-Lag-compensation.patch @@ -0,0 +1,216 @@ +From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001 +From: HaHaWTH <102713261+HaHaWTH@users.noreply.github.com> +Date: Mon, 1 Nov 2077 00:00:00 +0800 +Subject: [PATCH] TT20 Lag compensation + +This patch was ported from project: https://github.com/snackbag/TT20 +Project license: AGPL-3.0 + +diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java +index 09e55f62b4cea5b058e04356252f4f56957646b8..279289e0c3dc65fed0ee4b2f5b0ef077ab433d5e 100644 +--- a/src/main/java/net/minecraft/server/MinecraftServer.java ++++ b/src/main/java/net/minecraft/server/MinecraftServer.java +@@ -1628,6 +1628,12 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop 0 ? newTicks : 1; ++ else return newTicks; ++ } ++ ++ public static int tt20(int ticks, boolean limitZero) { ++ int newTicks = (int) Math.ceil(rawTT20(ticks)); ++ ++ if (limitZero) return newTicks > 0 ? newTicks : 1; ++ else return newTicks; ++ } ++ ++ public static double tt20(double ticks, boolean limitZero) { ++ double newTicks = rawTT20(ticks); ++ ++ if (limitZero) return newTicks > 0 ? newTicks : 1; ++ else return newTicks; ++ } ++ ++ public static double rawTT20(double ticks) { ++ return ticks == 0 ? 0 : ticks * TPSCalculator.getMostAccurateTPS() / TPSCalculator.MAX_TPS; ++ } ++ ++ public static class TPSCalculator { ++ public static Long lastTick; ++ public static Long currentTick; ++ private static double allMissedTicks = 0; ++ private static final List tpsHistory = Collections.synchronizedList(new DoubleArrayList()); ++ private static final int historyLimit = 40; ++ ++ public static final int MAX_TPS = 20; ++ public static final int FULL_TICK = 50; ++ ++ private TPSCalculator() {} ++ ++ public static void onTick() { ++ if (currentTick != null) { ++ lastTick = currentTick; ++ } ++ ++ currentTick = System.currentTimeMillis(); ++ addToHistory(getTPS()); ++ clearMissedTicks(); ++ missedTick(); ++ } ++ ++ private static void addToHistory(double tps) { ++ if (tpsHistory.size() >= historyLimit) { ++ tpsHistory.removeFirst(); ++ } ++ ++ tpsHistory.add(tps); ++ } ++ ++ public static long getMSPT() { ++ return currentTick - lastTick; ++ } ++ ++ public static double getAverageTPS() { ++ double sum = 0.0; ++ for (double value : tpsHistory) { ++ sum += value; ++ } ++ return tpsHistory.isEmpty() ? 0.1 : sum / tpsHistory.size(); ++ } ++ ++ public static double getTPS() { ++ if (lastTick == null) return -1; ++ if (getMSPT() <= 0) return 0.1; ++ ++ double tps = 1000 / (double) getMSPT(); ++ return tps > MAX_TPS ? MAX_TPS : tps; ++ } ++ ++ public static void missedTick() { ++ if (lastTick == null) return; ++ ++ long mspt = getMSPT() <= 0 ? 50 : getMSPT(); ++ double missedTicks = (mspt / (double) FULL_TICK) - 1; ++ allMissedTicks += missedTicks <= 0 ? 0 : missedTicks; ++ } ++ ++ public static double getMostAccurateTPS() { ++ return Math.min(getTPS(), getAverageTPS()); ++ } ++ ++ public double getAllMissedTicks() { ++ return allMissedTicks; ++ } ++ ++ public static int applicableMissedTicks() { ++ return (int) Math.floor(allMissedTicks); ++ } ++ ++ public static void clearMissedTicks() { ++ allMissedTicks -= applicableMissedTicks(); ++ } ++ ++ public void resetMissedTicks() { ++ allMissedTicks = 0; ++ } ++ } ++} +\ No newline at end of file From 6d03fd8926a22fbe640fedc35f5664f87e64b449 Mon Sep 17 00:00:00 2001 From: HaHaWTH <102713261+HaHaWTH@users.noreply.github.com> Date: Sat, 2 Nov 2024 15:12:37 +0800 Subject: [PATCH 6/7] Move to misc --- patches/server/0125-TT20-Lag-compensation.patch | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/patches/server/0125-TT20-Lag-compensation.patch b/patches/server/0125-TT20-Lag-compensation.patch index 61ea14003..9234aca00 100644 --- a/patches/server/0125-TT20-Lag-compensation.patch +++ b/patches/server/0125-TT20-Lag-compensation.patch @@ -63,7 +63,7 @@ index 9dcdb2f4001115db0c26fdbf86531dbe6098485d..53a28a4026c50a66e53241ffe660b4d7 @Override diff --git a/src/main/java/org/dreeam/leaf/config/modules/misc/LagCompensation.java b/src/main/java/org/dreeam/leaf/config/modules/misc/LagCompensation.java new file mode 100644 -index 0000000000000000000000000000000000000000..caeb9db146ea24d99d953d99d8e4ba725dcb2b3e +index 0000000000000000000000000000000000000000..d90ab4cb8c69318f8bf77af12e2840fb5d519931 --- /dev/null +++ b/src/main/java/org/dreeam/leaf/config/modules/misc/LagCompensation.java @@ -0,0 +1,26 @@ @@ -75,7 +75,7 @@ index 0000000000000000000000000000000000000000..caeb9db146ea24d99d953d99d8e4ba72 +public class LagCompensation extends ConfigModules { + + public String getBasePath() { -+ return EnumConfigCategory.PERF.getBaseKeyName() + ".lag-compensation"; ++ return EnumConfigCategory.MISC.getBaseKeyName() + ".lag-compensation"; + } + + public static boolean enabled = false; From cff7d959f436958b2908250fe53a9195d3b7e8a8 Mon Sep 17 00:00:00 2001 From: HaHaWTH <102713261+HaHaWTH@users.noreply.github.com> Date: Fri, 8 Nov 2024 02:01:24 +0800 Subject: [PATCH 7/7] Fix --- .../server/0119-Multithreaded-Tracker.patch | 4 +- ...patch => 0126-TT20-Lag-compensation.patch} | 2 +- ....patch => 0127-Asynchronous-locator.patch} | 4 +- ...28-Virtual-thread-for-chat-executor.patch} | 4 +- ...ckSort-in-NearestLivingEntitySensor.patch} | 0 ...uce-memory-footprint-of-CompoundTag.patch} | 0 ... 0131-Optimize-Entity-distanceToSqr.patch} | 0 ...e-snapshots-for-TileEntity-getOwner.patch} | 0 ...ault-don-t-use-blockstate-snapshots.patch} | 0 ... => 0134-Cache-tile-entity-position.patch} | 0 .../server/0134-TT20-Lag-compensation.patch | 216 ------------------ 11 files changed, 7 insertions(+), 223 deletions(-) rename patches/server/{0125-TT20-Lag-compensation.patch => 0126-TT20-Lag-compensation.patch} (98%) rename patches/server/{0126-Asynchronous-locator.patch => 0127-Asynchronous-locator.patch} (99%) rename patches/server/{0127-Virtual-thread-for-chat-executor.patch => 0128-Virtual-thread-for-chat-executor.patch} (94%) rename patches/server/{0128-Use-QuickSort-in-NearestLivingEntitySensor.patch => 0129-Use-QuickSort-in-NearestLivingEntitySensor.patch} (100%) rename patches/server/{0129-Further-reduce-memory-footprint-of-CompoundTag.patch => 0130-Further-reduce-memory-footprint-of-CompoundTag.patch} (100%) rename patches/server/{0130-Optimize-Entity-distanceToSqr.patch => 0131-Optimize-Entity-distanceToSqr.patch} (100%) rename patches/server/{0131-EMC-Don-t-use-snapshots-for-TileEntity-getOwner.patch => 0132-EMC-Don-t-use-snapshots-for-TileEntity-getOwner.patch} (100%) rename patches/server/{0132-EMC-Default-don-t-use-blockstate-snapshots.patch => 0133-EMC-Default-don-t-use-blockstate-snapshots.patch} (100%) rename patches/server/{0133-Cache-tile-entity-position.patch => 0134-Cache-tile-entity-position.patch} (100%) delete mode 100644 patches/server/0134-TT20-Lag-compensation.patch diff --git a/patches/server/0119-Multithreaded-Tracker.patch b/patches/server/0119-Multithreaded-Tracker.patch index 38d7848d9..2e65976b1 100644 --- a/patches/server/0119-Multithreaded-Tracker.patch +++ b/patches/server/0119-Multithreaded-Tracker.patch @@ -23,7 +23,7 @@ But it is still recommending to use those packet based, virtual entity based NPC plugins, e.g. ZNPC Plus, Adyeshach, Fancy NPC, etc. diff --git a/src/main/java/io/papermc/paper/plugin/manager/PaperEventManager.java b/src/main/java/io/papermc/paper/plugin/manager/PaperEventManager.java -index e42677bb004201efe1702779a78cc8d0ca05e80f..3fa04bad6f5263c626c22ef0a795947391e1e710 100644 +index e42677bb004201efe1702779a78cc8d0ca05e80f..6676be8304e9415099ed423d3315180cafebd928 100644 --- a/src/main/java/io/papermc/paper/plugin/manager/PaperEventManager.java +++ b/src/main/java/io/papermc/paper/plugin/manager/PaperEventManager.java @@ -42,6 +42,12 @@ class PaperEventManager { @@ -31,7 +31,7 @@ index e42677bb004201efe1702779a78cc8d0ca05e80f..3fa04bad6f5263c626c22ef0a7959473 throw new IllegalStateException(event.getEventName() + " may only be triggered asynchronously."); } else if (!event.isAsynchronous() && !this.server.isPrimaryThread() && !this.server.isStopping()) { + // Leaf start - Multithreaded tracker -+ if (org.dreeam.leaf.config.modules.async.MultithreadedTracker.enabled && Thread.currentThread() instanceof org.dreeam.leaf.async.tracker.MultithreadedTracker.MultithreadedTrackerThread) { ++ if (org.dreeam.leaf.config.modules.async.MultithreadedTracker.enabled) { + net.minecraft.server.MinecraftServer.getServer().scheduleOnMain(event::callEvent); + return; + } diff --git a/patches/server/0125-TT20-Lag-compensation.patch b/patches/server/0126-TT20-Lag-compensation.patch similarity index 98% rename from patches/server/0125-TT20-Lag-compensation.patch rename to patches/server/0126-TT20-Lag-compensation.patch index 9234aca00..44b3ea96e 100644 --- a/patches/server/0125-TT20-Lag-compensation.patch +++ b/patches/server/0126-TT20-Lag-compensation.patch @@ -7,7 +7,7 @@ This patch was ported from project: https://github.com/snackbag/TT20 Project license: AGPL-3.0 diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java -index 09e55f62b4cea5b058e04356252f4f56957646b8..279289e0c3dc65fed0ee4b2f5b0ef077ab433d5e 100644 +index 37c377761b45eefab39164a7e714e585a02a4447..f268001f7a821ad1d0db8a02de19e80cc951d0fd 100644 --- a/src/main/java/net/minecraft/server/MinecraftServer.java +++ b/src/main/java/net/minecraft/server/MinecraftServer.java @@ -1628,6 +1628,12 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop -Date: Mon, 1 Nov 2077 00:00:00 +0800 -Subject: [PATCH] TT20 Lag compensation - -This patch was ported from project: https://github.com/snackbag/TT20 -Project license: AGPL-3.0 - -diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java -index 09e55f62b4cea5b058e04356252f4f56957646b8..279289e0c3dc65fed0ee4b2f5b0ef077ab433d5e 100644 ---- a/src/main/java/net/minecraft/server/MinecraftServer.java -+++ b/src/main/java/net/minecraft/server/MinecraftServer.java -@@ -1628,6 +1628,12 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop 0 ? newTicks : 1; -+ else return newTicks; -+ } -+ -+ public static int tt20(int ticks, boolean limitZero) { -+ int newTicks = (int) Math.ceil(rawTT20(ticks)); -+ -+ if (limitZero) return newTicks > 0 ? newTicks : 1; -+ else return newTicks; -+ } -+ -+ public static double tt20(double ticks, boolean limitZero) { -+ double newTicks = rawTT20(ticks); -+ -+ if (limitZero) return newTicks > 0 ? newTicks : 1; -+ else return newTicks; -+ } -+ -+ public static double rawTT20(double ticks) { -+ return ticks == 0 ? 0 : ticks * TPSCalculator.getMostAccurateTPS() / TPSCalculator.MAX_TPS; -+ } -+ -+ public static class TPSCalculator { -+ public static Long lastTick; -+ public static Long currentTick; -+ private static double allMissedTicks = 0; -+ private static final List tpsHistory = Collections.synchronizedList(new DoubleArrayList()); -+ private static final int historyLimit = 40; -+ -+ public static final int MAX_TPS = 20; -+ public static final int FULL_TICK = 50; -+ -+ private TPSCalculator() {} -+ -+ public static void onTick() { -+ if (currentTick != null) { -+ lastTick = currentTick; -+ } -+ -+ currentTick = System.currentTimeMillis(); -+ addToHistory(getTPS()); -+ clearMissedTicks(); -+ missedTick(); -+ } -+ -+ private static void addToHistory(double tps) { -+ if (tpsHistory.size() >= historyLimit) { -+ tpsHistory.removeFirst(); -+ } -+ -+ tpsHistory.add(tps); -+ } -+ -+ public static long getMSPT() { -+ return currentTick - lastTick; -+ } -+ -+ public static double getAverageTPS() { -+ double sum = 0.0; -+ for (double value : tpsHistory) { -+ sum += value; -+ } -+ return tpsHistory.isEmpty() ? 0.1 : sum / tpsHistory.size(); -+ } -+ -+ public static double getTPS() { -+ if (lastTick == null) return -1; -+ if (getMSPT() <= 0) return 0.1; -+ -+ double tps = 1000 / (double) getMSPT(); -+ return tps > MAX_TPS ? MAX_TPS : tps; -+ } -+ -+ public static void missedTick() { -+ if (lastTick == null) return; -+ -+ long mspt = getMSPT() <= 0 ? 50 : getMSPT(); -+ double missedTicks = (mspt / (double) FULL_TICK) - 1; -+ allMissedTicks += missedTicks <= 0 ? 0 : missedTicks; -+ } -+ -+ public static double getMostAccurateTPS() { -+ return Math.min(getTPS(), getAverageTPS()); -+ } -+ -+ public double getAllMissedTicks() { -+ return allMissedTicks; -+ } -+ -+ public static int applicableMissedTicks() { -+ return (int) Math.floor(allMissedTicks); -+ } -+ -+ public static void clearMissedTicks() { -+ allMissedTicks -= applicableMissedTicks(); -+ } -+ -+ public void resetMissedTicks() { -+ allMissedTicks = 0; -+ } -+ } -+} -\ No newline at end of file