2020-05-06 17:48:49 +08:00
|
|
|
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
2018-07-04 15:55:24 +08:00
|
|
|
From: Aikar <aikar@aikar.co>
|
|
|
|
Date: Wed, 4 Jul 2018 02:10:36 -0400
|
|
|
|
Subject: [PATCH] Store reference to current Chunk for Entity and Block
|
|
|
|
Entities
|
|
|
|
|
|
|
|
This enables us a fast reference to the entities current chunk instead
|
|
|
|
of having to look it up by hashmap lookups.
|
|
|
|
|
|
|
|
diff --git a/src/main/java/net/minecraft/server/Chunk.java b/src/main/java/net/minecraft/server/Chunk.java
|
2020-05-25 23:12:22 +08:00
|
|
|
index 9397b58f047c837f8a9146723e4cd9a4d6b787a7..e1ac0c479ebfa2da69575db2032dd1415c4a41a5 100644
|
2018-07-04 15:55:24 +08:00
|
|
|
--- a/src/main/java/net/minecraft/server/Chunk.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/Chunk.java
|
2019-05-26 09:56:30 +08:00
|
|
|
@@ -29,7 +29,7 @@ public class Chunk implements IChunkAccess {
|
2019-04-24 09:00:24 +08:00
|
|
|
private final ChunkSection[] sections;
|
2019-12-11 08:56:03 +08:00
|
|
|
private BiomeStorage d;
|
2019-04-24 09:00:24 +08:00
|
|
|
private final Map<BlockPosition, NBTTagCompound> e;
|
|
|
|
- public boolean loaded;
|
|
|
|
+ public boolean loaded; public boolean isLoaded() { return loaded; } // Paper - OBFHELPER
|
2018-07-06 06:39:43 +08:00
|
|
|
public final World world;
|
2018-07-15 09:53:17 +08:00
|
|
|
public final Map<HeightMap.Type, HeightMap> heightMap;
|
2019-04-24 09:00:24 +08:00
|
|
|
private final ChunkConverter i;
|
2019-05-26 09:56:30 +08:00
|
|
|
@@ -55,11 +55,36 @@ public class Chunk implements IChunkAccess {
|
2019-12-11 08:56:03 +08:00
|
|
|
this(world, chunkcoordintpair, biomestorage, ChunkConverter.a, TickListEmpty.b(), TickListEmpty.b(), 0L, (ChunkSection[]) null, (Consumer) null);
|
2019-05-14 10:20:58 +08:00
|
|
|
}
|
2019-04-24 09:00:24 +08:00
|
|
|
|
2018-07-04 15:55:24 +08:00
|
|
|
+ // Paper start
|
|
|
|
+ private class TileEntityHashMap extends java.util.HashMap<BlockPosition, TileEntity> {
|
|
|
|
+ @Override
|
|
|
|
+ public TileEntity put(BlockPosition key, TileEntity value) {
|
|
|
|
+ TileEntity replaced = super.put(key, value);
|
|
|
|
+ if (replaced != null) {
|
|
|
|
+ replaced.setCurrentChunk(null);
|
|
|
|
+ }
|
|
|
|
+ if (value != null) {
|
|
|
|
+ value.setCurrentChunk(Chunk.this);
|
|
|
|
+ }
|
|
|
|
+ return replaced;
|
|
|
|
+ }
|
2019-04-24 09:00:24 +08:00
|
|
|
+
|
2018-07-04 15:55:24 +08:00
|
|
|
+ @Override
|
|
|
|
+ public TileEntity remove(Object key) {
|
|
|
|
+ TileEntity removed = super.remove(key);
|
|
|
|
+ if (removed != null) {
|
|
|
|
+ removed.setCurrentChunk(null);
|
|
|
|
+ }
|
|
|
|
+ return removed;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ // Paper end
|
2019-04-24 09:00:24 +08:00
|
|
|
+
|
2019-12-11 08:56:03 +08:00
|
|
|
public Chunk(World world, ChunkCoordIntPair chunkcoordintpair, BiomeStorage biomestorage, ChunkConverter chunkconverter, TickList<Block> ticklist, TickList<FluidType> ticklist1, long i, @Nullable ChunkSection[] achunksection, @Nullable Consumer<Chunk> consumer) {
|
2019-04-24 09:00:24 +08:00
|
|
|
this.sections = new ChunkSection[16];
|
|
|
|
this.e = Maps.newHashMap();
|
2018-10-23 07:16:21 +08:00
|
|
|
this.heightMap = Maps.newEnumMap(HeightMap.Type.class);
|
2018-07-04 15:55:24 +08:00
|
|
|
- this.tileEntities = Maps.newHashMap();
|
|
|
|
+ this.tileEntities = new TileEntityHashMap(); // Paper
|
2019-04-24 09:00:24 +08:00
|
|
|
this.l = Maps.newHashMap();
|
|
|
|
this.m = Maps.newHashMap();
|
|
|
|
this.n = new ShortList[16];
|
2020-05-25 23:12:22 +08:00
|
|
|
@@ -465,6 +490,7 @@ public class Chunk implements IChunkAccess {
|
2019-04-06 11:08:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
entity.inChunk = true;
|
|
|
|
+ entity.setCurrentChunk(this); // Paper
|
2019-04-24 09:00:24 +08:00
|
|
|
entity.chunkX = this.loc.x;
|
2018-12-17 13:18:06 +08:00
|
|
|
entity.chunkY = k;
|
2019-04-24 09:00:24 +08:00
|
|
|
entity.chunkZ = this.loc.z;
|
2020-05-25 23:12:22 +08:00
|
|
|
@@ -477,6 +503,7 @@ public class Chunk implements IChunkAccess {
|
2019-04-06 11:08:45 +08:00
|
|
|
((HeightMap) this.heightMap.get(heightmap_type)).a(along);
|
2018-07-04 15:55:24 +08:00
|
|
|
}
|
|
|
|
|
2019-04-06 11:08:45 +08:00
|
|
|
+ public void removeEntity(Entity entity) { this.b(entity); } // Paper - OBFHELPER
|
|
|
|
public void b(Entity entity) {
|
|
|
|
this.a(entity, entity.chunkY);
|
|
|
|
}
|
2020-05-25 23:12:22 +08:00
|
|
|
@@ -490,7 +517,12 @@ public class Chunk implements IChunkAccess {
|
2018-07-15 09:53:17 +08:00
|
|
|
i = this.entitySlices.length - 1;
|
2018-07-04 15:55:24 +08:00
|
|
|
}
|
2020-02-01 00:09:56 +08:00
|
|
|
|
2018-07-15 09:53:17 +08:00
|
|
|
- this.entitySlices[i].remove(entity);
|
2018-07-04 15:55:24 +08:00
|
|
|
+ // Paper start
|
2019-04-06 11:08:45 +08:00
|
|
|
+ if (entity.currentChunk != null && entity.currentChunk.get() == this) entity.setCurrentChunk(null);
|
2018-07-15 09:53:17 +08:00
|
|
|
+ if (!this.entitySlices[i].remove(entity)) {
|
|
|
|
+ return;
|
|
|
|
+ }
|
2018-07-04 15:55:24 +08:00
|
|
|
+ // Paper end
|
2020-02-01 00:09:56 +08:00
|
|
|
this.entities.remove(entity); // Paper
|
2018-07-04 15:55:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
diff --git a/src/main/java/net/minecraft/server/Entity.java b/src/main/java/net/minecraft/server/Entity.java
|
2020-04-22 23:11:50 +08:00
|
|
|
index 6c7816fdc1ef476969cb70c8ea697b60746d6ab4..0d8262b1d2722e716f95db4c9a8a132c54b613cf 100644
|
2018-07-04 15:55:24 +08:00
|
|
|
--- a/src/main/java/net/minecraft/server/Entity.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/Entity.java
|
2019-12-11 08:56:03 +08:00
|
|
|
@@ -134,7 +134,7 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, Ke
|
2019-04-24 09:00:24 +08:00
|
|
|
private static final DataWatcherObject<Boolean> aB = DataWatcher.a(Entity.class, DataWatcherRegistry.i);
|
|
|
|
private static final DataWatcherObject<Boolean> aC = DataWatcher.a(Entity.class, DataWatcherRegistry.i);
|
2019-05-14 10:20:58 +08:00
|
|
|
protected static final DataWatcherObject<EntityPose> POSE = DataWatcher.a(Entity.class, DataWatcherRegistry.s);
|
2018-07-30 01:12:39 +08:00
|
|
|
- public boolean inChunk;
|
|
|
|
+ public boolean inChunk; public boolean isAddedToChunk() { return inChunk; } // Paper - OBFHELPER
|
2018-12-17 13:18:06 +08:00
|
|
|
public int chunkX; public int getChunkX() { return chunkX; } // Paper - OBFHELPER
|
|
|
|
public int chunkY; public int getChunkY() { return chunkY; } // Paper - OBFHELPER
|
|
|
|
public int chunkZ; public int getChunkZ() { return chunkZ; } // Paper - OBFHELPER
|
2020-04-02 12:43:11 +08:00
|
|
|
@@ -178,7 +178,7 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, Ke
|
|
|
|
}
|
|
|
|
|
|
|
|
public boolean isChunkLoaded() {
|
|
|
|
- return world.isChunkLoaded((int) Math.floor(this.locX) >> 4, (int) Math.floor(this.locZ) >> 4);
|
|
|
|
+ return getCurrentChunk() != null;
|
|
|
|
}
|
|
|
|
// CraftBukkit end
|
|
|
|
|
2020-04-22 23:11:50 +08:00
|
|
|
@@ -1709,6 +1709,43 @@ public abstract class Entity implements INamableTileEntity, ICommandListener, Ke
|
2018-07-04 15:55:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Paper start
|
2019-04-06 11:08:45 +08:00
|
|
|
+ java.lang.ref.WeakReference<Chunk> currentChunk = null;
|
2018-07-14 12:12:42 +08:00
|
|
|
+
|
|
|
|
+ public void setCurrentChunk(Chunk chunk) {
|
|
|
|
+ this.currentChunk = chunk != null ? new java.lang.ref.WeakReference<>(chunk) : null;
|
|
|
|
+ }
|
|
|
|
+ /**
|
|
|
|
+ * Returns the entities current registered chunk. If the entity is not added to a chunk yet, it will return null
|
|
|
|
+ */
|
2018-07-04 15:55:24 +08:00
|
|
|
+ public Chunk getCurrentChunk() {
|
2018-07-09 10:39:46 +08:00
|
|
|
+ final Chunk chunk = currentChunk != null ? currentChunk.get() : null;
|
2020-04-02 12:43:11 +08:00
|
|
|
+ if (chunk != null && chunk.isLoaded()) {
|
|
|
|
+ return chunk;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return !isAddedToChunk() ? null : ((ChunkProviderServer) world.chunkProvider).getChunkAtIfLoadedMainThreadNoCache(getChunkX(), getChunkZ());
|
2018-07-04 15:55:24 +08:00
|
|
|
+ }
|
2020-04-02 12:43:11 +08:00
|
|
|
+
|
2018-07-14 12:12:42 +08:00
|
|
|
+ /**
|
|
|
|
+ * Returns the chunk at the location, using the entities local cache if avail
|
|
|
|
+ * Will only return null if the location specified is not loaded
|
|
|
|
+ */
|
2018-07-13 04:42:56 +08:00
|
|
|
+ public Chunk getCurrentChunkAt(int x, int z) {
|
2020-04-02 12:43:11 +08:00
|
|
|
+ Chunk chunk = getCurrentChunk();
|
|
|
|
+ if (chunk != null && getChunkX() == chunk.getPos().x && getChunkZ() == chunk.getPos().z) {
|
|
|
|
+ return chunk;
|
2018-07-13 04:42:56 +08:00
|
|
|
+ }
|
2020-04-02 12:43:11 +08:00
|
|
|
+ return ((ChunkProviderServer) world.chunkProvider).getChunkAtIfLoadedMainThreadNoCache(getChunkX(), getChunkZ());
|
2018-07-13 04:42:56 +08:00
|
|
|
+ }
|
2018-07-14 12:12:42 +08:00
|
|
|
+ /**
|
|
|
|
+ * Returns the chunk at the entities current location, using the entities local cache if avail
|
2020-04-02 12:43:11 +08:00
|
|
|
+ * Ideally this is always the same as getCurrentChunk, but only becomes different in registration issues.
|
2018-07-14 12:12:42 +08:00
|
|
|
+ * Will only return null if the location specified is not loaded
|
|
|
|
+ */
|
2018-07-13 04:42:56 +08:00
|
|
|
+ public Chunk getChunkAtLocation() {
|
|
|
|
+ return getCurrentChunkAt((int)Math.floor(locX) >> 4, (int)Math.floor(locZ) >> 4);
|
2018-07-04 15:55:24 +08:00
|
|
|
+ }
|
2018-07-15 09:53:17 +08:00
|
|
|
+
|
2018-07-27 12:44:53 +08:00
|
|
|
private MinecraftKey entityKey;
|
|
|
|
private String entityKeyString;
|
2018-07-04 15:55:24 +08:00
|
|
|
|
|
|
|
diff --git a/src/main/java/net/minecraft/server/TileEntity.java b/src/main/java/net/minecraft/server/TileEntity.java
|
2020-05-06 17:48:49 +08:00
|
|
|
index 9071bb7ece6d38d85b4ea11e53671f3ec5210b28..820180ab3f7053c348caa80cc21f15dfa3d26afd 100644
|
2018-07-04 15:55:24 +08:00
|
|
|
--- a/src/main/java/net/minecraft/server/TileEntity.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/TileEntity.java
|
2019-05-14 10:20:58 +08:00
|
|
|
@@ -51,6 +51,15 @@ public abstract class TileEntity implements KeyedObject { // Paper
|
2018-07-27 12:44:53 +08:00
|
|
|
getMinecraftKey(); // Try to load if it doesn't exists.
|
2018-07-15 09:53:17 +08:00
|
|
|
return tileEntityKeyString;
|
2018-07-04 15:55:24 +08:00
|
|
|
}
|
2018-07-15 09:53:17 +08:00
|
|
|
+
|
2018-07-04 15:55:24 +08:00
|
|
|
+ private java.lang.ref.WeakReference<Chunk> currentChunk = null;
|
|
|
|
+ public Chunk getCurrentChunk() {
|
2018-07-27 12:44:53 +08:00
|
|
|
+ final Chunk chunk = currentChunk != null ? currentChunk.get() : null;
|
2018-07-06 06:39:43 +08:00
|
|
|
+ return chunk != null && chunk.isLoaded() ? chunk : null;
|
2018-07-04 15:55:24 +08:00
|
|
|
+ }
|
|
|
|
+ public void setCurrentChunk(Chunk chunk) {
|
|
|
|
+ this.currentChunk = chunk != null ? new java.lang.ref.WeakReference<>(chunk) : null;
|
|
|
|
+ }
|
2018-07-15 09:53:17 +08:00
|
|
|
// Paper end
|
2018-07-04 15:55:24 +08:00
|
|
|
|
2018-07-15 09:53:17 +08:00
|
|
|
@Nullable
|
2018-07-04 15:55:24 +08:00
|
|
|
diff --git a/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java b/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java
|
2020-06-02 11:15:47 +08:00
|
|
|
index 27e2379f2dd6e9e06ae7a973ab8654cdcacf66c0..9672dd7c2166bba88832b1743b97e16561fe434c 100644
|
2018-07-04 15:55:24 +08:00
|
|
|
--- a/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java
|
|
|
|
+++ b/src/main/java/org/bukkit/craftbukkit/entity/CraftEntity.java
|
2020-06-02 11:15:47 +08:00
|
|
|
@@ -137,6 +137,7 @@ import net.minecraft.server.EntityZombieHusk;
|
|
|
|
import net.minecraft.server.EntityZombieVillager;
|
2019-04-24 09:00:24 +08:00
|
|
|
import net.minecraft.server.IChatBaseComponent;
|
|
|
|
import net.minecraft.server.NBTTagCompound;
|
|
|
|
+import org.bukkit.Chunk; // Paper
|
2018-07-04 15:55:24 +08:00
|
|
|
import org.bukkit.EntityEffect;
|
|
|
|
import org.bukkit.Location;
|
|
|
|
import org.bukkit.Server;
|
2020-06-02 11:15:47 +08:00
|
|
|
@@ -178,6 +179,12 @@ public abstract class CraftEntity implements org.bukkit.entity.Entity {
|
2018-07-04 15:55:24 +08:00
|
|
|
this.entity = entity;
|
|
|
|
}
|
|
|
|
|
|
|
|
+ @Override
|
|
|
|
+ public Chunk getChunk() {
|
|
|
|
+ net.minecraft.server.Chunk currentChunk = entity.getCurrentChunk();
|
|
|
|
+ return currentChunk != null ? currentChunk.bukkitChunk : getLocation().getChunk();
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
public static CraftEntity getEntity(CraftServer server, Entity entity) {
|
2020-04-27 15:34:45 +08:00
|
|
|
/*
|
2018-07-04 15:55:24 +08:00
|
|
|
* Order is *EXTREMELY* important -- keep it right! =D
|