2019-12-13 22:08:40 +08:00
|
|
|
From eec5541d4e4b72b627794f842de2b26d94865944 Mon Sep 17 00:00:00 2001
|
2015-06-16 20:55:15 +08:00
|
|
|
From: Byteflux <byte@byteflux.net>
|
2016-03-01 07:09:49 +08:00
|
|
|
Date: Wed, 2 Mar 2016 11:59:48 -0600
|
2015-06-16 20:55:15 +08:00
|
|
|
Subject: [PATCH] Optimize explosions
|
|
|
|
|
|
|
|
The process of determining an entity's exposure from explosions can be
|
|
|
|
expensive when there are hundreds or more entities in range.
|
|
|
|
|
|
|
|
This patch adds a per-tick cache that is used for storing and retrieving
|
|
|
|
an entity's exposure during an explosion.
|
|
|
|
|
2016-03-01 07:09:49 +08:00
|
|
|
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
2019-12-11 08:56:03 +08:00
|
|
|
index 50dec5cb5..f038d3f7d 100644
|
2016-03-01 07:09:49 +08:00
|
|
|
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
|
|
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
2019-10-27 06:55:58 +08:00
|
|
|
@@ -150,4 +150,10 @@ public class PaperWorldConfig {
|
2018-07-15 09:53:17 +08:00
|
|
|
disableEndCredits = getBoolean("game-mechanics.disable-end-credits", false);
|
|
|
|
log("End credits disabled: " + disableEndCredits);
|
2016-03-01 07:09:49 +08:00
|
|
|
}
|
|
|
|
+
|
|
|
|
+ public boolean optimizeExplosions;
|
|
|
|
+ private void optimizeExplosions() {
|
|
|
|
+ optimizeExplosions = getBoolean("optimize-explosions", false);
|
|
|
|
+ log("Optimize explosions: " + optimizeExplosions);
|
|
|
|
+ }
|
|
|
|
}
|
2015-06-16 20:55:15 +08:00
|
|
|
diff --git a/src/main/java/net/minecraft/server/Explosion.java b/src/main/java/net/minecraft/server/Explosion.java
|
2019-12-13 22:08:40 +08:00
|
|
|
index d2c31ac1f..64cf5a29c 100644
|
2015-06-16 20:55:15 +08:00
|
|
|
--- a/src/main/java/net/minecraft/server/Explosion.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/Explosion.java
|
2019-12-11 08:56:03 +08:00
|
|
|
@@ -177,7 +177,7 @@ public class Explosion {
|
2015-06-16 20:55:15 +08:00
|
|
|
d8 /= d11;
|
|
|
|
d9 /= d11;
|
|
|
|
d10 /= d11;
|
2019-04-24 10:34:11 +08:00
|
|
|
- double d12 = (double) a(vec3d, entity);
|
|
|
|
+ double d12 = this.getBlockDensity(vec3d, entity); // Paper - Optimize explosions
|
2015-06-16 20:55:15 +08:00
|
|
|
double d13 = (1.0D - d7) * d12;
|
|
|
|
|
2016-03-01 07:09:49 +08:00
|
|
|
// CraftBukkit start
|
2019-12-13 22:08:40 +08:00
|
|
|
@@ -384,4 +384,84 @@ public class Explosion {
|
2019-04-24 10:34:11 +08:00
|
|
|
|
|
|
|
private Effect() {}
|
2015-06-16 20:55:15 +08:00
|
|
|
}
|
2016-03-01 07:09:49 +08:00
|
|
|
+ // Paper start - Optimize explosions
|
2019-04-24 10:34:11 +08:00
|
|
|
+ private float getBlockDensity(Vec3D vec3d, Entity entity) {
|
2016-03-01 07:09:49 +08:00
|
|
|
+ if (!this.world.paperConfig.optimizeExplosions) {
|
2019-04-24 10:34:11 +08:00
|
|
|
+ return a(vec3d, entity);
|
2015-06-16 20:55:15 +08:00
|
|
|
+ }
|
2019-04-24 10:34:11 +08:00
|
|
|
+ CacheKey key = new CacheKey(this, entity.getBoundingBox());
|
2015-06-16 20:55:15 +08:00
|
|
|
+ Float blockDensity = this.world.explosionDensityCache.get(key);
|
|
|
|
+ if (blockDensity == null) {
|
2019-04-24 10:34:11 +08:00
|
|
|
+ blockDensity = a(vec3d, entity);
|
2015-06-16 20:55:15 +08:00
|
|
|
+ this.world.explosionDensityCache.put(key, blockDensity);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return blockDensity;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ static class CacheKey {
|
|
|
|
+ private final World world;
|
|
|
|
+ private final double posX, posY, posZ;
|
|
|
|
+ private final double minX, minY, minZ;
|
|
|
|
+ private final double maxX, maxY, maxZ;
|
|
|
|
+
|
|
|
|
+ public CacheKey(Explosion explosion, AxisAlignedBB aabb) {
|
|
|
|
+ this.world = explosion.world;
|
|
|
|
+ this.posX = explosion.posX;
|
|
|
|
+ this.posY = explosion.posY;
|
|
|
|
+ this.posZ = explosion.posZ;
|
2018-10-23 07:16:21 +08:00
|
|
|
+ this.minX = aabb.minX;
|
|
|
|
+ this.minY = aabb.minY;
|
|
|
|
+ this.minZ = aabb.minZ;
|
|
|
|
+ this.maxX = aabb.maxX;
|
|
|
|
+ this.maxY = aabb.maxY;
|
|
|
|
+ this.maxZ = aabb.maxZ;
|
2015-06-16 20:55:15 +08:00
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ @Override
|
|
|
|
+ public boolean equals(Object o) {
|
|
|
|
+ if (this == o) return true;
|
|
|
|
+ if (o == null || getClass() != o.getClass()) return false;
|
|
|
|
+
|
|
|
|
+ CacheKey cacheKey = (CacheKey) o;
|
|
|
|
+
|
|
|
|
+ if (Double.compare(cacheKey.posX, posX) != 0) return false;
|
|
|
|
+ if (Double.compare(cacheKey.posY, posY) != 0) return false;
|
|
|
|
+ if (Double.compare(cacheKey.posZ, posZ) != 0) return false;
|
|
|
|
+ if (Double.compare(cacheKey.minX, minX) != 0) return false;
|
|
|
|
+ if (Double.compare(cacheKey.minY, minY) != 0) return false;
|
|
|
|
+ if (Double.compare(cacheKey.minZ, minZ) != 0) return false;
|
|
|
|
+ if (Double.compare(cacheKey.maxX, maxX) != 0) return false;
|
|
|
|
+ if (Double.compare(cacheKey.maxY, maxY) != 0) return false;
|
|
|
|
+ if (Double.compare(cacheKey.maxZ, maxZ) != 0) return false;
|
|
|
|
+ return world.equals(cacheKey.world);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ @Override
|
|
|
|
+ public int hashCode() {
|
|
|
|
+ int result;
|
|
|
|
+ long temp;
|
|
|
|
+ result = world.hashCode();
|
|
|
|
+ temp = Double.doubleToLongBits(posX);
|
|
|
|
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
|
|
+ temp = Double.doubleToLongBits(posY);
|
|
|
|
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
|
|
+ temp = Double.doubleToLongBits(posZ);
|
|
|
|
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
|
|
+ temp = Double.doubleToLongBits(minX);
|
|
|
|
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
|
|
+ temp = Double.doubleToLongBits(minY);
|
|
|
|
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
|
|
+ temp = Double.doubleToLongBits(minZ);
|
|
|
|
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
|
|
+ temp = Double.doubleToLongBits(maxX);
|
|
|
|
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
|
|
+ temp = Double.doubleToLongBits(maxY);
|
|
|
|
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
|
|
+ temp = Double.doubleToLongBits(maxZ);
|
|
|
|
+ result = 31 * result + (int) (temp ^ (temp >>> 32));
|
|
|
|
+ return result;
|
|
|
|
+ }
|
|
|
|
+ }
|
2016-03-01 07:09:49 +08:00
|
|
|
+ // Paper end
|
2015-06-16 20:55:15 +08:00
|
|
|
}
|
|
|
|
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
|
2019-12-11 08:56:03 +08:00
|
|
|
index 983b6906e..81d3a5262 100644
|
2015-06-16 20:55:15 +08:00
|
|
|
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
|
2019-12-11 08:56:03 +08:00
|
|
|
@@ -1170,6 +1170,7 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
|
2019-04-24 10:34:11 +08:00
|
|
|
|
2018-12-17 13:18:06 +08:00
|
|
|
this.methodProfiler.exit();
|
|
|
|
this.methodProfiler.exit();
|
2016-03-01 07:09:49 +08:00
|
|
|
+ worldserver.explosionDensityCache.clear(); // Paper - Optimize explosions
|
2018-07-15 09:53:17 +08:00
|
|
|
}
|
2018-08-27 02:11:49 +08:00
|
|
|
}
|
2015-06-16 20:55:15 +08:00
|
|
|
|
|
|
|
diff --git a/src/main/java/net/minecraft/server/World.java b/src/main/java/net/minecraft/server/World.java
|
2019-12-13 22:08:40 +08:00
|
|
|
index 5793ebd00..b1f0c6597 100644
|
2015-06-16 20:55:15 +08:00
|
|
|
--- a/src/main/java/net/minecraft/server/World.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/World.java
|
2019-04-24 10:34:11 +08:00
|
|
|
@@ -19,6 +19,7 @@ import org.apache.logging.log4j.util.Supplier;
|
2016-03-25 14:38:38 +08:00
|
|
|
// CraftBukkit start
|
|
|
|
import com.google.common.collect.Maps;
|
2019-01-01 11:15:55 +08:00
|
|
|
import java.util.ArrayList;
|
2016-03-25 14:38:38 +08:00
|
|
|
+import java.util.HashMap; // Paper
|
|
|
|
import java.util.Map;
|
|
|
|
import org.bukkit.Bukkit;
|
|
|
|
import org.bukkit.block.BlockState;
|
2019-12-11 08:56:03 +08:00
|
|
|
@@ -93,6 +94,7 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
2016-03-25 14:38:38 +08:00
|
|
|
private org.spigotmc.TickLimiter entityLimiter;
|
2015-09-13 12:29:23 +08:00
|
|
|
private org.spigotmc.TickLimiter tileLimiter;
|
2015-06-16 20:55:15 +08:00
|
|
|
private int tileTickPosition;
|
2016-03-25 14:38:38 +08:00
|
|
|
+ public final Map<Explosion.CacheKey, Float> explosionDensityCache = new HashMap<>(); // Paper - Optimize explosions
|
2015-06-16 20:55:15 +08:00
|
|
|
|
2016-03-01 07:09:49 +08:00
|
|
|
public CraftWorld getWorld() {
|
|
|
|
return this.world;
|
2015-06-16 20:55:15 +08:00
|
|
|
--
|
2019-12-13 22:08:40 +08:00
|
|
|
2.24.1
|
2015-06-16 20:55:15 +08:00
|
|
|
|