2018-10-06 04:56:20 +00:00
|
|
|
From 3b7f8f6ddc6df313b6a5434a1ff7692dbe91c828 Mon Sep 17 00:00:00 2001
|
2016-02-14 01:54:01 +00:00
|
|
|
From: CullanP <cullanpage@gmail.com>
|
2016-02-29 23:09:49 +00:00
|
|
|
Date: Thu, 3 Mar 2016 02:13:38 -0600
|
2016-02-14 01:54:01 +00:00
|
|
|
Subject: [PATCH] Avoid hopper searches if there are no items
|
|
|
|
|
|
|
|
Hoppers searching for items and minecarts is the most expensive part of hopper ticking.
|
|
|
|
We keep track of the number of minecarts and items in a chunk.
|
|
|
|
If there are no items in the chunk, we skip searching for items.
|
|
|
|
If there are no minecarts in the chunk, we skip searching for them.
|
|
|
|
|
|
|
|
Usually hoppers aren't near items, so we can skip most item searches.
|
|
|
|
And since minecart hoppers are used _very_ rarely near we can avoid alot of searching there.
|
|
|
|
|
|
|
|
Combined, this adds up a lot.
|
|
|
|
|
|
|
|
diff --git a/src/main/java/net/minecraft/server/Chunk.java b/src/main/java/net/minecraft/server/Chunk.java
|
2018-10-06 04:56:20 +00:00
|
|
|
index 6fa379e580..4a0be58e2f 100644
|
2016-02-14 01:54:01 +00:00
|
|
|
--- a/src/main/java/net/minecraft/server/Chunk.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/Chunk.java
|
2018-08-26 18:11:49 +00:00
|
|
|
@@ -91,6 +91,10 @@ public class Chunk implements IChunkAccess {
|
2018-07-15 01:53:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
final PaperLightingQueue.LightingQueue lightingQueue = new PaperLightingQueue.LightingQueue(this);
|
2016-02-29 23:09:49 +00:00
|
|
|
+ // Track the number of minecarts and items
|
2016-02-14 01:54:01 +00:00
|
|
|
+ // Keep this synced with entitySlices.add() and entitySlices.remove()
|
|
|
|
+ private final int[] itemCounts = new int[16];
|
|
|
|
+ private final int[] inventoryEntityCounts = new int[16];
|
2018-07-15 01:53:17 +00:00
|
|
|
// Paper end
|
|
|
|
public boolean areNeighborsLoaded(final int radius) {
|
|
|
|
switch (radius) {
|
2018-10-06 04:56:20 +00:00
|
|
|
@@ -684,6 +688,11 @@ public class Chunk implements IChunkAccess {
|
2018-07-15 01:53:17 +00:00
|
|
|
entity.ag = this.locZ;
|
2016-02-14 01:54:01 +00:00
|
|
|
this.entitySlices[k].add(entity);
|
2018-07-15 01:53:17 +00:00
|
|
|
// Paper start
|
2016-02-14 01:54:01 +00:00
|
|
|
+ if (entity instanceof EntityItem) {
|
|
|
|
+ itemCounts[k]++;
|
|
|
|
+ } else if (entity instanceof IInventory) {
|
|
|
|
+ inventoryEntityCounts[k]++;
|
|
|
|
+ }
|
2018-07-15 01:53:17 +00:00
|
|
|
entity.setCurrentChunk(this);
|
2018-07-27 04:44:53 +00:00
|
|
|
entityCounts.increment(entity.getMinecraftKeyString());
|
2018-07-15 01:53:17 +00:00
|
|
|
// Paper end
|
2018-10-06 04:56:20 +00:00
|
|
|
@@ -709,6 +718,11 @@ public class Chunk implements IChunkAccess {
|
2018-07-15 01:53:17 +00:00
|
|
|
if (!this.entitySlices[i].remove(entity)) {
|
|
|
|
return;
|
2016-02-14 01:54:01 +00:00
|
|
|
}
|
|
|
|
+ if (entity instanceof EntityItem) {
|
|
|
|
+ itemCounts[i]--;
|
|
|
|
+ } else if (entity instanceof IInventory) {
|
|
|
|
+ inventoryEntityCounts[i]--;
|
|
|
|
+ }
|
2018-07-15 01:53:17 +00:00
|
|
|
entity.setCurrentChunk(null);
|
2018-07-27 04:44:53 +00:00
|
|
|
entityCounts.decrement(entity.getMinecraftKeyString());
|
2018-07-15 01:53:17 +00:00
|
|
|
// Paper end
|
2018-10-06 04:56:20 +00:00
|
|
|
@@ -948,6 +962,15 @@ public class Chunk implements IChunkAccess {
|
2016-02-14 01:54:01 +00:00
|
|
|
if (!this.entitySlices[k].isEmpty()) {
|
|
|
|
Iterator iterator = this.entitySlices[k].iterator();
|
2016-02-29 23:09:49 +00:00
|
|
|
|
|
|
|
+ // Paper start - Don't search for inventories if we have none, and that is all we want
|
2016-02-14 01:54:01 +00:00
|
|
|
+ /*
|
2016-02-29 23:09:49 +00:00
|
|
|
+ * We check if they want inventories by seeing if it is the static `IEntitySelector.c`
|
|
|
|
+ *
|
|
|
|
+ * Make sure the inventory selector stays in sync.
|
|
|
|
+ * It should be the one that checks `var1 instanceof IInventory && var1.isAlive()`
|
|
|
|
+ */
|
2016-02-14 01:54:01 +00:00
|
|
|
+ if (predicate == IEntitySelector.c && inventoryEntityCounts[k] <= 0) continue;
|
2016-02-29 23:09:49 +00:00
|
|
|
+ // Paper end
|
2016-02-14 01:54:01 +00:00
|
|
|
while (iterator.hasNext()) {
|
|
|
|
Entity entity1 = (Entity) iterator.next();
|
|
|
|
|
2018-10-06 04:56:20 +00:00
|
|
|
@@ -984,7 +1007,18 @@ public class Chunk implements IChunkAccess {
|
2016-02-14 01:54:01 +00:00
|
|
|
i = MathHelper.clamp(i, 0, this.entitySlices.length - 1);
|
|
|
|
j = MathHelper.clamp(j, 0, this.entitySlices.length - 1);
|
|
|
|
|
2016-02-29 23:09:49 +00:00
|
|
|
+ // Paper start
|
2016-02-14 01:54:01 +00:00
|
|
|
+ int[] counts;
|
2016-05-06 00:22:01 +00:00
|
|
|
+ if (EntityItem.class.isAssignableFrom(oclass)) {
|
2016-02-14 01:54:01 +00:00
|
|
|
+ counts = itemCounts;
|
|
|
|
+ } else if (IInventory.class.isAssignableFrom(oclass)) {
|
|
|
|
+ counts = inventoryEntityCounts;
|
|
|
|
+ } else {
|
|
|
|
+ counts = null;
|
|
|
|
+ }
|
2016-02-29 23:09:49 +00:00
|
|
|
+ // Paper end
|
2016-02-14 01:54:01 +00:00
|
|
|
for (int k = i; k <= j; ++k) {
|
2016-02-29 23:09:49 +00:00
|
|
|
+ if (counts != null && counts[k] <= 0) continue; // Paper - Don't check a chunk if it doesn't have the type we are looking for
|
2016-02-14 01:54:01 +00:00
|
|
|
Iterator iterator = this.entitySlices[k].iterator(); // Spigot
|
|
|
|
|
|
|
|
while (iterator.hasNext()) {
|
|
|
|
--
|
Improve Light Queue and force enable it for all
There is no reason for the light queue to even be an option. This
enables the light queue for everyone.
This also improves the "can we still tick" time logic to always
check before running a light operation.
previously, we always executed at least 10 on the first world
(but not other worlds...), but we are seeing light take up some
heavy time, so improving that for now.
I've now also improved recheck gaps logic to happen at the end of all single block updates
This also prevents multiple gap checks, as previously if a tick skipped
the gaps check, the next tick would end up re-adding the entry again,
resulting in multiple gap checks.
This now just sets a marker "We need to recheck gaps" and will only occur
once.
This also should reduce chunk loads, as previously, we checked if
the neighbor chunks were loaded for the gap check, however those
neighbor chunks might of unloaded before the light queue operation
actually ran. Now, the neighbor chunk is done when the gap check
is being done, so it should avoid loading chunks.
Fixes #1466
Fixes #1431
2018-09-22 15:46:31 +00:00
|
|
|
2.19.0
|
2016-02-14 01:54:01 +00:00
|
|
|
|