2015-05-25 12:37:24 +02:00
|
|
|
--- a/net/minecraft/server/ChunkProviderServer.java
|
|
|
|
+++ b/net/minecraft/server/ChunkProviderServer.java
|
2016-05-11 07:34:16 +02:00
|
|
|
@@ -14,10 +14,16 @@
|
2014-11-25 22:32:16 +01:00
|
|
|
import org.apache.logging.log4j.LogManager;
|
|
|
|
import org.apache.logging.log4j.Logger;
|
|
|
|
|
|
|
|
+// CraftBukkit start
|
|
|
|
+import org.bukkit.Server;
|
|
|
|
+import org.bukkit.craftbukkit.chunkio.ChunkIOExecutor;
|
|
|
|
+import org.bukkit.event.world.ChunkUnloadEvent;
|
|
|
|
+// CraftBukkit end
|
|
|
|
+
|
|
|
|
public class ChunkProviderServer implements IChunkProvider {
|
|
|
|
|
2016-02-29 22:32:46 +01:00
|
|
|
private static final Logger a = LogManager.getLogger();
|
2016-05-10 13:47:39 +02:00
|
|
|
- private final Set<Long> unloadQueue = Sets.newHashSet();
|
2016-05-11 07:34:16 +02:00
|
|
|
+ public final it.unimi.dsi.fastutil.longs.LongSet unloadQueue = new it.unimi.dsi.fastutil.longs.LongArraySet(); // CraftBukkit Set -> LongSet
|
2016-05-10 13:47:39 +02:00
|
|
|
public final ChunkGenerator chunkGenerator;
|
2016-02-29 22:32:46 +01:00
|
|
|
private final IChunkLoader chunkLoader;
|
2016-05-10 13:47:39 +02:00
|
|
|
public final Long2ObjectMap<Chunk> chunks = new Long2ObjectOpenHashMap(8192);
|
2016-05-11 07:34:16 +02:00
|
|
|
@@ -69,19 +75,68 @@
|
2016-02-29 22:32:46 +01:00
|
|
|
Chunk chunk = this.getLoadedChunkAt(i, j);
|
|
|
|
|
|
|
|
if (chunk == null) {
|
|
|
|
- chunk = this.loadChunk(i, j);
|
|
|
|
+ // CraftBukkit start
|
|
|
|
+ ChunkRegionLoader loader = null;
|
|
|
|
+
|
|
|
|
+ if (this.chunkLoader instanceof ChunkRegionLoader) {
|
|
|
|
+ loader = (ChunkRegionLoader) this.chunkLoader;
|
|
|
|
+ }
|
|
|
|
+ if (loader != null && loader.chunkExists(world, i, j)) {
|
|
|
|
+ chunk = ChunkIOExecutor.syncChunkLoad(world, loader, this, i, j);
|
|
|
|
+ }
|
|
|
|
+ /* chunk = this.loadChunk(i, j);
|
|
|
|
if (chunk != null) {
|
|
|
|
this.chunks.put(ChunkCoordIntPair.a(i, j), chunk);
|
|
|
|
chunk.addEntities();
|
|
|
|
chunk.loadNearby(this, this.chunkGenerator);
|
|
|
|
}
|
|
|
|
+ */
|
|
|
|
+ // CraftBukkit end
|
|
|
|
}
|
|
|
|
|
|
|
|
return chunk;
|
|
|
|
}
|
|
|
|
|
|
|
|
public Chunk getChunkAt(int i, int j) {
|
2014-11-25 22:32:16 +01:00
|
|
|
+ return getChunkAt(i, j, null);
|
|
|
|
+ }
|
2015-02-26 23:41:06 +01:00
|
|
|
+
|
2014-11-25 22:32:16 +01:00
|
|
|
+ public Chunk getChunkAt(int i, int j, Runnable runnable) {
|
2016-04-20 12:00:35 +02:00
|
|
|
+ return getChunkAt(i, j, runnable, true);
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ public Chunk getChunkAt(int i, int j, Runnable runnable, boolean generate) {
|
2016-05-10 13:47:39 +02:00
|
|
|
+ Chunk chunk = chunks.get(ChunkCoordIntPair.a(i, j));
|
2014-11-25 22:32:16 +01:00
|
|
|
+ ChunkRegionLoader loader = null;
|
2015-02-26 23:41:06 +01:00
|
|
|
+
|
2014-11-25 22:32:16 +01:00
|
|
|
+ if (this.chunkLoader instanceof ChunkRegionLoader) {
|
|
|
|
+ loader = (ChunkRegionLoader) this.chunkLoader;
|
2016-02-29 22:32:46 +01:00
|
|
|
+
|
2014-11-25 22:32:16 +01:00
|
|
|
+ }
|
|
|
|
+ // We can only use the queue for already generated chunks
|
|
|
|
+ if (chunk == null && loader != null && loader.chunkExists(world, i, j)) {
|
|
|
|
+ if (runnable != null) {
|
|
|
|
+ ChunkIOExecutor.queueChunkLoad(world, loader, this, i, j, runnable);
|
|
|
|
+ return null;
|
|
|
|
+ } else {
|
|
|
|
+ chunk = ChunkIOExecutor.syncChunkLoad(world, loader, this, i, j);
|
|
|
|
+ }
|
2016-04-20 12:00:35 +02:00
|
|
|
+ } else if (chunk == null && generate) {
|
2014-11-25 22:32:16 +01:00
|
|
|
+ chunk = originalGetChunkAt(i, j);
|
|
|
|
+ }
|
2015-02-26 23:41:06 +01:00
|
|
|
+
|
2014-11-25 22:32:16 +01:00
|
|
|
+ // If we didn't load the chunk async and have a callback run it now
|
|
|
|
+ if (runnable != null) {
|
|
|
|
+ runnable.run();
|
|
|
|
+ }
|
2015-02-26 23:41:06 +01:00
|
|
|
+
|
2014-11-25 22:32:16 +01:00
|
|
|
+ return chunk;
|
|
|
|
+ }
|
2016-02-29 22:32:46 +01:00
|
|
|
+
|
2014-11-25 22:32:16 +01:00
|
|
|
+ public Chunk originalGetChunkAt(int i, int j) {
|
2016-05-10 13:47:39 +02:00
|
|
|
Chunk chunk = this.getOrLoadChunkAt(i, j);
|
2014-11-25 22:32:16 +01:00
|
|
|
+ boolean newChunk = false;
|
|
|
|
+ // CraftBukkit end
|
|
|
|
|
|
|
|
if (chunk == null) {
|
2016-02-29 22:32:46 +01:00
|
|
|
long k = ChunkCoordIntPair.a(i, j);
|
2016-05-11 07:34:16 +02:00
|
|
|
@@ -97,9 +152,37 @@
|
2016-05-10 13:47:39 +02:00
|
|
|
crashreportsystemdetails.a("Generator", (Object) this.chunkGenerator);
|
|
|
|
throw new ReportedException(crashreport);
|
2014-11-25 22:32:16 +01:00
|
|
|
}
|
2016-05-10 13:47:39 +02:00
|
|
|
+ newChunk = true; // CraftBukkit
|
2014-11-25 22:32:16 +01:00
|
|
|
|
2016-05-10 13:47:39 +02:00
|
|
|
this.chunks.put(k, chunk);
|
2014-11-25 22:32:16 +01:00
|
|
|
chunk.addEntities();
|
2016-02-29 22:32:46 +01:00
|
|
|
+
|
2014-11-25 22:32:16 +01:00
|
|
|
+ // CraftBukkit start
|
|
|
|
+ Server server = world.getServer();
|
|
|
|
+ if (server != null) {
|
|
|
|
+ /*
|
|
|
|
+ * If it's a new world, the first few chunks are generated inside
|
|
|
|
+ * the World constructor. We can't reliably alter that, so we have
|
|
|
|
+ * no way of creating a CraftWorld/CraftServer at that point.
|
|
|
|
+ */
|
|
|
|
+ server.getPluginManager().callEvent(new org.bukkit.event.world.ChunkLoadEvent(chunk.bukkitChunk, newChunk));
|
|
|
|
+ }
|
2015-02-26 23:41:06 +01:00
|
|
|
+
|
2014-11-25 22:32:16 +01:00
|
|
|
+ // Update neighbor counts
|
|
|
|
+ for (int x = -2; x < 3; x++) {
|
|
|
|
+ for (int z = -2; z < 3; z++) {
|
|
|
|
+ if (x == 0 && z == 0) {
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+
|
2016-05-10 13:47:39 +02:00
|
|
|
+ Chunk neighbor = this.getLoadedChunkAt(chunk.locX + x, chunk.locZ + z);
|
2014-11-25 22:32:16 +01:00
|
|
|
+ if (neighbor != null) {
|
|
|
|
+ neighbor.setNeighborLoaded(-x, -z);
|
|
|
|
+ chunk.setNeighborLoaded(x, z);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ // CraftBukkit end
|
2016-02-29 22:32:46 +01:00
|
|
|
chunk.loadNearby(this, this.chunkGenerator);
|
2014-11-25 22:32:16 +01:00
|
|
|
}
|
|
|
|
|
2016-05-11 07:34:16 +02:00
|
|
|
@@ -146,10 +229,12 @@
|
2015-02-26 23:41:06 +01:00
|
|
|
|
2016-02-29 22:32:46 +01:00
|
|
|
public boolean a(boolean flag) {
|
2014-11-25 22:32:16 +01:00
|
|
|
int i = 0;
|
2016-05-10 13:47:39 +02:00
|
|
|
- ArrayList arraylist = Lists.newArrayList(this.chunks.values());
|
2015-02-26 23:41:06 +01:00
|
|
|
|
|
|
|
- for (int j = 0; j < arraylist.size(); ++j) {
|
|
|
|
- Chunk chunk = (Chunk) arraylist.get(j);
|
2014-11-25 22:32:16 +01:00
|
|
|
+ // CraftBukkit start
|
|
|
|
+ Iterator iterator = this.chunks.values().iterator();
|
|
|
|
+ while (iterator.hasNext()) {
|
|
|
|
+ Chunk chunk = (Chunk) iterator.next();
|
|
|
|
+ // CraftBukkit end
|
|
|
|
|
|
|
|
if (flag) {
|
|
|
|
this.saveChunkNOP(chunk);
|
2016-05-11 07:34:16 +02:00
|
|
|
@@ -182,6 +267,29 @@
|
|
|
|
Chunk chunk = (Chunk) this.chunks.get(olong);
|
|
|
|
|
|
|
|
if (chunk != null && chunk.d) {
|
|
|
|
+ // CraftBukkit start
|
|
|
|
+ ChunkUnloadEvent event = new ChunkUnloadEvent(chunk.bukkitChunk);
|
|
|
|
+ this.world.getServer().getPluginManager().callEvent(event);
|
|
|
|
+ if (event.isCancelled()) {
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // Update neighbor counts
|
|
|
|
+ for (int x = -2; x < 3; x++) {
|
|
|
|
+ for (int z = -2; z < 3; z++) {
|
|
|
|
+ if (x == 0 && z == 0) {
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ Chunk neighbor = this.getLoadedChunkAt(chunk.locX + x, chunk.locZ + z);
|
|
|
|
+ if (neighbor != null) {
|
|
|
|
+ neighbor.setNeighborUnloaded(-x, -z);
|
|
|
|
+ chunk.setNeighborUnloaded(x, z);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ // CraftBukkit end
|
2014-11-25 22:32:16 +01:00
|
|
|
+
|
|
|
|
chunk.removeEntities();
|
|
|
|
this.saveChunk(chunk);
|
|
|
|
this.saveChunkNOP(chunk);
|