From f481fac7c9a15fdc169a5184c466759f203b9e2f Mon Sep 17 00:00:00 2001 From: Aikar Date: Fri, 12 Oct 2018 01:29:28 -0400 Subject: [PATCH] Improvements to DataFixers for performance Mojang asked me to make some changes, so applied them here. --- ...and-performance-issues-in-DataFixers.patch | 74 ++++++------------- ...taFixer-Bootstrap-to-prebuild-faster.patch | 63 ++++++++++++++++ scripts/importmcdev.sh | 1 + 3 files changed, 85 insertions(+), 53 deletions(-) create mode 100644 Spigot-Server-Patches/Optimize-DataFixer-Bootstrap-to-prebuild-faster.patch diff --git a/Spigot-Server-Patches/Fix-concurrency-and-performance-issues-in-DataFixers.patch b/Spigot-Server-Patches/Fix-concurrency-and-performance-issues-in-DataFixers.patch index cf387e36af..087e91536a 100644 --- a/Spigot-Server-Patches/Fix-concurrency-and-performance-issues-in-DataFixers.patch +++ b/Spigot-Server-Patches/Fix-concurrency-and-performance-issues-in-DataFixers.patch @@ -75,52 +75,38 @@ index 2c259d74e9..17481fb6e6 100644 return Either.right(new Type.FieldNotFoundException(String.format("Type error for choice type \"%s\": expected type: %s, actual type: %s)", name, targetType, elementType))); } diff --git a/src/main/java/com/mojang/datafixers/functions/PointFree.java b/src/main/java/com/mojang/datafixers/functions/PointFree.java -index 0d88490f77..028942b8ea 100644 +index 0d88490f77..4d2af956a0 100644 --- a/src/main/java/com/mojang/datafixers/functions/PointFree.java +++ b/src/main/java/com/mojang/datafixers/functions/PointFree.java -@@ -0,0 +0,0 @@ public abstract class PointFree { +@@ -0,0 +0,0 @@ import java.util.Optional; + import java.util.function.Function; + + public abstract class PointFree { +- private boolean initialized; ++ private volatile boolean initialized; + @Nullable private Function, T> value; @SuppressWarnings("ConstantConditions") -- public Function, T> evalCached() { -+ public synchronized Function, T> evalCached() { // Paper + public Function, T> evalCached() { if (!initialized) { - initialized = true; - value = eval(); +- initialized = true; +- value = eval(); ++ synchronized (this) { ++ if (!initialized) { ++ value = eval(); ++ initialized = true; ++ } ++ } + } + return value; + } diff --git a/src/main/java/com/mojang/datafixers/schemas/Schema.java b/src/main/java/com/mojang/datafixers/schemas/Schema.java -index 7c67d989e0..7faca88a88 100644 +index 7c67d989e0..d81e8d1a1a 100644 --- a/src/main/java/com/mojang/datafixers/schemas/Schema.java +++ b/src/main/java/com/mojang/datafixers/schemas/Schema.java -@@ -0,0 +0,0 @@ import java.util.function.Function; - import java.util.function.Supplier; - - public class Schema { -- protected final Object2IntMap RECURSIVE_TYPES = new Object2IntOpenHashMap<>(); -- private final Map> TYPE_TEMPLATES = Maps.newHashMap(); -+ protected final Object2IntMap RECURSIVE_TYPES = it.unimi.dsi.fastutil.objects.Object2IntMaps.synchronize(new Object2IntOpenHashMap<>()); // Paper -+ private final Map> TYPE_TEMPLATES = Maps.newConcurrentMap(); // Paper - private final Map> TYPES; - private final int versionKey; - private final String name; @@ -0,0 +0,0 @@ public class Schema { - } - protected Map> buildTypes() { -- final Map> types = Maps.newHashMap(); -+ final Map> types = Maps.newConcurrentMap(); // Paper - - final List templates = Lists.newArrayList(); - -+ synchronized (RECURSIVE_TYPES) { // Paper - for (final Object2IntMap.Entry entry : RECURSIVE_TYPES.object2IntEntrySet()) { - templates.add(DSL.check(entry.getKey(), entry.getIntValue(), getTemplate(entry.getKey()))); -- } -+ } } // Paper - - final TypeTemplate choice = templates.stream().reduce(DSL::or).get(); - final TypeFamily family = new RecursiveTypeFamily(name, choice); - -+ synchronized (TYPE_TEMPLATES) { // Paper for (final String name : TYPE_TEMPLATES.keySet()) { final Type type; - if (RECURSIVE_TYPES.containsKey(name)) { @@ -133,12 +119,6 @@ index 7c67d989e0..7faca88a88 100644 } else { type = getTemplate(name).apply(family).apply(-1); } - types.put(name, type); -- } -+ } } // Paper - return types; - } - @@ -0,0 +0,0 @@ public class Schema { } @@ -153,18 +133,6 @@ index 7c67d989e0..7faca88a88 100644 } return getTemplate(name); } -@@ -0,0 +0,0 @@ public class Schema { - public void registerType(final boolean recursive, final DSL.TypeReference type, final Supplier template) { - TYPE_TEMPLATES.put(type.typeName(), template); - // TODO: calculate recursiveness instead of hardcoding -+ synchronized (RECURSIVE_TYPES) { // Paper - if (recursive && !RECURSIVE_TYPES.containsKey(type.typeName())) { - RECURSIVE_TYPES.put(type.typeName(), RECURSIVE_TYPES.size()); -- } -+ } } // Paper - } - - public int getVersionKey() { diff --git a/src/main/java/com/mojang/datafixers/types/DynamicOps.java b/src/main/java/com/mojang/datafixers/types/DynamicOps.java index 3c76929f24..f21531b3cd 100644 --- a/src/main/java/com/mojang/datafixers/types/DynamicOps.java diff --git a/Spigot-Server-Patches/Optimize-DataFixer-Bootstrap-to-prebuild-faster.patch b/Spigot-Server-Patches/Optimize-DataFixer-Bootstrap-to-prebuild-faster.patch new file mode 100644 index 0000000000..0545f0de02 --- /dev/null +++ b/Spigot-Server-Patches/Optimize-DataFixer-Bootstrap-to-prebuild-faster.patch @@ -0,0 +1,63 @@ +From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001 +From: Aikar +Date: Fri, 12 Oct 2018 01:28:32 -0400 +Subject: [PATCH] Optimize DataFixer Bootstrap to prebuild faster + + +diff --git a/src/main/java/com/mojang/datafixers/DataFixerBuilder.java b/src/main/java/com/mojang/datafixers/DataFixerBuilder.java +index 8bc1e3b976..114f958601 100644 +--- a/src/main/java/com/mojang/datafixers/DataFixerBuilder.java ++++ b/src/main/java/com/mojang/datafixers/DataFixerBuilder.java +@@ -0,0 +0,0 @@ import org.apache.logging.log4j.LogManager; + import org.apache.logging.log4j.Logger; + + import java.util.ArrayList; ++import java.util.Collections; + import java.util.List; ++import java.util.Set; + import java.util.concurrent.Executor; + import java.util.function.BiFunction; + +@@ -0,0 +0,0 @@ public class DataFixerBuilder { + final DataFixerUpper fixerUpper = new DataFixerUpper(new Int2ObjectAVLTreeMap<>(schemas), new ArrayList<>(globalList), new IntAVLTreeSet(fixerVersions)); + + executor.execute(() -> { +- final IntBidirectionalIterator iterator = fixerUpper.fixerVersions().iterator(); ++ List taskList = new ArrayList<>(); ++ IntBidirectionalIterator iterator = fixerUpper.fixerVersions().iterator(); + while (iterator.hasNext()) { + final int versionKey = iterator.nextInt(); + final Schema schema = schemas.get(versionKey); ++ + for (final String typeName: schema.types()) { +- final Type dataType = schema.getType(() -> typeName); +- final TypeRewriteRule rule = fixerUpper.getRule(DataFixUtils.getVersion(versionKey), dataVersion); +- dataType.rewrite(rule, DataFixerUpper.OPTIMIZATION_RULE); ++ taskList.add(() -> { ++ final Type dataType = schema.getType(() -> typeName); ++ final TypeRewriteRule rule = fixerUpper.getRule(DataFixUtils.getVersion(versionKey), dataVersion); ++ dataType.rewrite(rule, DataFixerUpper.OPTIMIZATION_RULE); ++ }); + } + } ++ ++ // Divide up into sets of tasks by number of CPU cores ++ // Some tasks are faster than others, randomize it to try to divide it more ++ Collections.shuffle(taskList); ++ List> buildQueue = new ArrayList<>(); ++ List current = new ArrayList<>(); ++ buildQueue.add(current); ++ int maxTasks = (int) Math.max(1, Math.floor(taskList.size() / (float)Math.min(6, Runtime.getRuntime().availableProcessors()-2))); ++ for (Runnable task : taskList) { ++ if (current.size() > maxTasks) { ++ current = new ArrayList<>(); ++ buildQueue.add(current); ++ } ++ current.add(task); ++ } ++ ++ buildQueue.forEach(queue -> executor.execute(() -> queue.forEach(Runnable::run))); + }); + + return fixerUpper; +-- \ No newline at end of file diff --git a/scripts/importmcdev.sh b/scripts/importmcdev.sh index dbfda717fe..c893fd2850 100755 --- a/scripts/importmcdev.sh +++ b/scripts/importmcdev.sh @@ -104,6 +104,7 @@ done # # group # lib # prefix # many files importLibrary com.mojang datafixerupper com/mojang/datafixers \ schemas/Schema.java \ + DataFixerBuilder.java \ DataFixerUpper.java \ NamedChoiceFinder.java \ functions/PointFree.java \