2021-11-24 11:06:34 +00:00
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
2022-09-26 08:02:51 +00:00
From: Spottedleaf <Spottedleaf@users.noreply.github.com>
2021-11-24 11:06:34 +00:00
Date: Mon, 6 Apr 2020 17:53:29 -0700
Subject: [PATCH] Optimize GoalSelector Goal.Flag Set operations
Optimise the stream.anyMatch statement to move to a bitset
where we can replace the call with a single bitwise operation.
diff --git a/src/main/java/net/minecraft/world/entity/ai/goal/Goal.java b/src/main/java/net/minecraft/world/entity/ai/goal/Goal.java
2024-01-23 17:01:39 +00:00
index 16f9a98b8a939e5ca7e2dc04f87134a7ed66736b..dd423302b1baa64ef86ded87a29659342e28c142 100644
2021-11-24 11:06:34 +00:00
--- a/src/main/java/net/minecraft/world/entity/ai/goal/Goal.java
+++ b/src/main/java/net/minecraft/world/entity/ai/goal/Goal.java
2024-01-23 11:06:27 +00:00
@@ -4,7 +4,16 @@ import java.util.EnumSet;
2021-11-24 11:06:34 +00:00
import net.minecraft.util.Mth;
public abstract class Goal {
- private final EnumSet<Goal.Flag> flags = EnumSet.noneOf(Goal.Flag.class);
+ private final EnumSet<Goal.Flag> flags = EnumSet.noneOf(Goal.Flag.class); // Paper unused, but dummy to prevent plugins from crashing as hard. Theyll need to support paper in a special case if this is super important, but really doesn't seem like it would be.
+ private final com.destroystokyo.paper.util.set.OptimizedSmallEnumSet<net.minecraft.world.entity.ai.goal.Goal.Flag> goalTypes = new com.destroystokyo.paper.util.set.OptimizedSmallEnumSet<>(Goal.Flag.class); // Paper - remove streams from pathfindergoalselector
2024-01-23 11:06:27 +00:00
+
+ // Paper start - remove streams from pathfindergoalselector; make sure types are not empty
+ public Goal() {
+ if (this.goalTypes.size() == 0) {
2024-01-23 17:01:39 +00:00
+ this.goalTypes.add(Flag.UNKNOWN_BEHAVIOR);
2024-01-23 11:06:27 +00:00
+ }
+ }
+ // Paper end - remove streams from pathfindergoalselector
2021-11-24 11:06:34 +00:00
public abstract boolean canUse();
2024-01-23 11:06:27 +00:00
@@ -30,8 +39,13 @@ public abstract class Goal {
2021-11-24 11:06:34 +00:00
}
public void setFlags(EnumSet<Goal.Flag> controls) {
- this.flags.clear();
- this.flags.addAll(controls);
+ // Paper start - remove streams from pathfindergoalselector
+ this.goalTypes.clear();
2024-01-23 17:01:39 +00:00
+ this.goalTypes.addAll(controls);
2024-01-23 11:06:27 +00:00
+ if (this.goalTypes.size() == 0) {
2024-01-23 17:01:39 +00:00
+ this.goalTypes.add(Flag.UNKNOWN_BEHAVIOR);
2024-01-23 11:06:27 +00:00
+ }
2021-11-24 11:06:34 +00:00
+ // Paper end - remove streams from pathfindergoalselector
}
@Override
2024-01-23 11:06:27 +00:00
@@ -39,8 +53,10 @@ public abstract class Goal {
2021-11-24 11:06:34 +00:00
return this.getClass().getSimpleName();
}
- public EnumSet<Goal.Flag> getFlags() {
- return this.flags;
+ // Paper start - remove streams from pathfindergoalselector
+ public com.destroystokyo.paper.util.set.OptimizedSmallEnumSet<Goal.Flag> getFlags() {
+ return this.goalTypes;
+ // Paper end - remove streams from pathfindergoalselector
}
protected int adjustedTickDelay(int ticks) {
diff --git a/src/main/java/net/minecraft/world/entity/ai/goal/GoalSelector.java b/src/main/java/net/minecraft/world/entity/ai/goal/GoalSelector.java
2024-04-12 19:14:06 +00:00
index 040d62effc651d14d3557f8ff582cb078b74ae1e..38af5c7280366fd6ec077f3d914ea5f3ee77451a 100644
2021-11-24 11:06:34 +00:00
--- a/src/main/java/net/minecraft/world/entity/ai/goal/GoalSelector.java
+++ b/src/main/java/net/minecraft/world/entity/ai/goal/GoalSelector.java
2024-04-12 19:14:06 +00:00
@@ -31,10 +31,12 @@ public class GoalSelector {
2021-11-24 11:06:34 +00:00
private final Map<Goal.Flag, WrappedGoal> lockedFlags = new EnumMap<>(Goal.Flag.class);
2022-11-19 23:53:20 +00:00
private final Set<WrappedGoal> availableGoals = Sets.newLinkedHashSet();
2021-11-24 11:06:34 +00:00
private final Supplier<ProfilerFiller> profiler;
- private final EnumSet<Goal.Flag> disabledFlags = EnumSet.noneOf(Goal.Flag.class);
+ private final EnumSet<Goal.Flag> disabledFlags = EnumSet.noneOf(Goal.Flag.class); // Paper unused, but dummy to prevent plugins from crashing as hard. Theyll need to support paper in a special case if this is super important, but really doesn't seem like it would be.
+ private final com.destroystokyo.paper.util.set.OptimizedSmallEnumSet<net.minecraft.world.entity.ai.goal.Goal.Flag> goalTypes = new com.destroystokyo.paper.util.set.OptimizedSmallEnumSet<>(Goal.Flag.class); // Paper - remove streams from pathfindergoalselector
private int tickCount;
private int newGoalRate = 3;
private int curRate;
+ private static final Goal.Flag[] GOAL_FLAG_VALUES = Goal.Flag.values(); // Paper - remove streams from pathfindergoalselector
public GoalSelector(Supplier<ProfilerFiller> profiler) {
this.profiler = profiler;
2024-04-12 19:14:06 +00:00
@@ -64,22 +66,32 @@ public class GoalSelector {
2021-11-24 11:06:34 +00:00
}
// Paper end
public void removeGoal(Goal goal) {
2024-04-12 19:14:06 +00:00
- this.availableGoals.stream().filter(wrappedGoal -> wrappedGoal.getGoal() == goal).filter(WrappedGoal::isRunning).forEach(WrappedGoal::stop);
- this.availableGoals.removeIf(wrappedGoal -> wrappedGoal.getGoal() == goal);
2023-03-23 21:57:03 +00:00
- }
-
- private static boolean goalContainsAnyFlags(WrappedGoal goal, EnumSet<Goal.Flag> controls) {
2024-04-12 19:14:06 +00:00
- for (Goal.Flag flag : goal.getFlags()) {
2023-03-23 21:57:03 +00:00
- if (controls.contains(flag)) {
- return true;
2021-11-24 11:06:34 +00:00
+ // Paper start - remove streams from pathfindergoalselector
+ for (java.util.Iterator<WrappedGoal> iterator = this.availableGoals.iterator(); iterator.hasNext();) {
+ WrappedGoal goalWrapped = iterator.next();
+ if (goalWrapped.getGoal() != goal) {
+ continue;
2023-03-23 21:57:03 +00:00
}
2021-11-24 11:06:34 +00:00
+ if (goalWrapped.isRunning()) {
+ goalWrapped.stop();
2021-11-30 08:36:30 +00:00
+ }
2021-11-24 11:06:34 +00:00
+ iterator.remove();
2023-03-23 21:57:03 +00:00
}
2021-11-24 11:06:34 +00:00
+ // Paper end - remove streams from pathfindergoalselector
2023-03-23 21:57:03 +00:00
+ }
2021-11-24 11:06:34 +00:00
- return false;
+ private static boolean goalContainsAnyFlags(WrappedGoal goal, com.destroystokyo.paper.util.set.OptimizedSmallEnumSet<Goal.Flag> controls) {
+ return goal.getFlags().hasCommonElements(controls); // Paper
}
private static boolean goalCanBeReplacedForAllFlags(WrappedGoal goal, Map<Goal.Flag, WrappedGoal> goalsByControl) {
2024-04-12 19:14:06 +00:00
- for (Goal.Flag flag : goal.getFlags()) {
2021-11-24 11:06:34 +00:00
+ // Paper start
2021-11-30 08:36:30 +00:00
+ long flagIterator = goal.getFlags().getBackingSet();
+ int wrappedGoalSize = goal.getFlags().size();
+ for (int i = 0; i < wrappedGoalSize; ++i) {
+ final Goal.Flag flag = GOAL_FLAG_VALUES[Long.numberOfTrailingZeros(flagIterator)];
+ flagIterator ^= io.papermc.paper.util.IntegerUtil.getTrailingBit(flagIterator);
+ // Paper end
2021-11-24 11:06:34 +00:00
if (!goalsByControl.getOrDefault(flag, NO_GOAL).canBeReplacedBy(goal)) {
2021-11-30 08:36:30 +00:00
return false;
2021-11-24 11:06:34 +00:00
}
2024-04-12 19:14:06 +00:00
@@ -93,7 +105,7 @@ public class GoalSelector {
2021-11-24 11:06:34 +00:00
profilerFiller.push("goalCleanup");
2024-04-12 19:14:06 +00:00
for (WrappedGoal wrappedGoal : this.availableGoals) {
2021-11-24 11:06:34 +00:00
- if (wrappedGoal.isRunning() && (goalContainsAnyFlags(wrappedGoal, this.disabledFlags) || !wrappedGoal.canContinueToUse())) {
2024-04-12 19:14:06 +00:00
+ if (wrappedGoal.isRunning() && (goalContainsAnyFlags(wrappedGoal, this.goalTypes) || !wrappedGoal.canContinueToUse())) { // Paper - Perf: optimize goal types by removing streams
2021-11-24 11:06:34 +00:00
wrappedGoal.stop();
}
}
2024-04-12 19:14:06 +00:00
@@ -111,11 +123,14 @@ public class GoalSelector {
2021-11-24 11:06:34 +00:00
profilerFiller.push("goalUpdate");
2024-04-12 19:14:06 +00:00
for (WrappedGoal wrappedGoal2 : this.availableGoals) {
- if (!wrappedGoal2.isRunning()
- && !goalContainsAnyFlags(wrappedGoal2, this.disabledFlags)
- && goalCanBeReplacedForAllFlags(wrappedGoal2, this.lockedFlags)
- && wrappedGoal2.canUse()) {
- for (Goal.Flag flag : wrappedGoal2.getFlags()) {
2021-11-24 11:06:34 +00:00
+ // Paper start
+ if (!wrappedGoal2.isRunning() && !goalContainsAnyFlags(wrappedGoal2, this.goalTypes) && goalCanBeReplacedForAllFlags(wrappedGoal2, this.lockedFlags) && wrappedGoal2.canUse()) {
2021-11-30 08:36:30 +00:00
+ long flagIterator = wrappedGoal2.getFlags().getBackingSet();
+ int wrappedGoalSize = wrappedGoal2.getFlags().size();
+ for (int i = 0; i < wrappedGoalSize; ++i) {
+ final Goal.Flag flag = GOAL_FLAG_VALUES[Long.numberOfTrailingZeros(flagIterator)];
+ flagIterator ^= io.papermc.paper.util.IntegerUtil.getTrailingBit(flagIterator);
2021-11-24 11:06:34 +00:00
+ // Paper end
WrappedGoal wrappedGoal3 = this.lockedFlags.getOrDefault(flag, NO_GOAL);
wrappedGoal3.stop();
this.lockedFlags.put(flag, wrappedGoal2);
2024-04-12 19:14:06 +00:00
@@ -155,11 +170,11 @@ public class GoalSelector {
2021-11-24 11:06:34 +00:00
}
public void disableControlFlag(Goal.Flag control) {
- this.disabledFlags.add(control);
2024-01-23 17:01:39 +00:00
+ this.goalTypes.add(control); // Paper - remove streams from pathfindergoalselector
2021-11-24 11:06:34 +00:00
}
public void enableControlFlag(Goal.Flag control) {
- this.disabledFlags.remove(control);
2024-01-23 17:01:39 +00:00
+ this.goalTypes.remove(control); // Paper - remove streams from pathfindergoalselector
2021-11-24 11:06:34 +00:00
}
public void setControlFlag(Goal.Flag control, boolean enabled) {
diff --git a/src/main/java/net/minecraft/world/entity/ai/goal/WrappedGoal.java b/src/main/java/net/minecraft/world/entity/ai/goal/WrappedGoal.java
2024-04-12 19:14:06 +00:00
index b02d3deb550830245c8945ef17d3073ea930fdda..65ccdbaa5230c02d44a5959bca0f6fc30237a6fd 100644
2021-11-24 11:06:34 +00:00
--- a/src/main/java/net/minecraft/world/entity/ai/goal/WrappedGoal.java
+++ b/src/main/java/net/minecraft/world/entity/ai/goal/WrappedGoal.java
@@ -69,8 +69,10 @@ public class WrappedGoal extends Goal {
}
@Override
- public EnumSet<Goal.Flag> getFlags() {
+ // Paper start - remove streams from pathfindergoalselector
+ public com.destroystokyo.paper.util.set.OptimizedSmallEnumSet<Goal.Flag> getFlags() {
return this.goal.getFlags();
+ // Paper end - remove streams from pathfindergoalselector
}
public boolean isRunning() {