2021-03-16 09:00:00 +11:00
--- a/net/minecraft/server/level/ChunkProviderServer.java
+++ b/net/minecraft/server/level/ChunkProviderServer.java
2024-06-14 01:05:00 +10:00
@@ -82,6 +82,16 @@
2020-06-06 19:23:46 +10:00
this.clearCache();
}
+ // CraftBukkit start - properly implement isChunkLoaded
+ public boolean isChunkLoaded(int chunkX, int chunkZ) {
2021-11-22 09:00:00 +11:00
+ PlayerChunk chunk = this.chunkMap.getUpdatingChunkIfPresent(ChunkCoordIntPair.asLong(chunkX, chunkZ));
2020-06-06 19:23:46 +10:00
+ if (chunk == null) {
+ return false;
+ }
2022-03-27 16:24:06 +11:00
+ return chunk.getFullChunkNow() != null;
2020-06-06 19:23:46 +10:00
+ }
+ // CraftBukkit end
+
@Override
public LightEngineThreaded getLightEngine() {
return this.lightEngine;
2024-06-14 01:05:00 +10:00
@@ -125,7 +135,7 @@
2021-06-11 15:00:00 +10:00
if (k == this.lastChunkPos[l] && chunkstatus == this.lastChunkStatus[l]) {
2024-04-24 01:15:00 +10:00
IChunkAccess ichunkaccess = this.lastChunk[l];
2019-05-16 01:11:20 +02:00
- if (ichunkaccess != null || !flag) {
+ if (ichunkaccess != null) { // CraftBukkit - the chunk can become accessible in the meantime TODO for non-null chunks it might also make sense to check that the chunk's state hasn't changed in the meantime
return ichunkaccess;
}
}
2024-06-14 01:05:00 +10:00
@@ -138,7 +148,7 @@
2024-04-24 01:15:00 +10:00
Objects.requireNonNull(completablefuture);
chunkproviderserver_b.managedBlock(completablefuture::isDone);
ChunkResult<IChunkAccess> chunkresult = (ChunkResult) completablefuture.join();
- IChunkAccess ichunkaccess1 = (IChunkAccess) chunkresult.orElse((Object) null);
+ IChunkAccess ichunkaccess1 = (IChunkAccess) chunkresult.orElse(null); // CraftBukkit - decompile error
if (ichunkaccess1 == null && flag) {
throw (IllegalStateException) SystemUtils.pauseInIde(new IllegalStateException("Chunk not there when requested: " + chunkresult.getError()));
2024-06-14 01:05:00 +10:00
@@ -218,7 +228,15 @@
2023-06-08 01:30:00 +10:00
int l = ChunkLevel.byStatus(chunkstatus);
2021-11-22 09:00:00 +11:00
PlayerChunk playerchunk = this.getVisibleChunkIfPresent(k);
2019-05-16 01:11:20 +02:00
- if (flag) {
+ // CraftBukkit start - don't add new ticket for currently unloading chunk
+ boolean currentlyUnloading = false;
+ if (playerchunk != null) {
2023-06-08 01:30:00 +10:00
+ FullChunkStatus oldChunkState = ChunkLevel.fullStatus(playerchunk.oldTicketLevel);
+ FullChunkStatus currentChunkState = ChunkLevel.fullStatus(playerchunk.getTicketLevel());
+ currentlyUnloading = (oldChunkState.isOrAfter(FullChunkStatus.FULL) && !currentChunkState.isOrAfter(FullChunkStatus.FULL));
2019-05-16 01:11:20 +02:00
+ }
+ if (flag && !currentlyUnloading) {
+ // CraftBukkit end
2021-11-22 09:00:00 +11:00
this.distanceManager.addTicket(TicketType.UNKNOWN, chunkcoordintpair, l, chunkcoordintpair);
if (this.chunkAbsent(playerchunk, l)) {
GameProfilerFiller gameprofilerfiller = this.level.getProfiler();
2024-06-14 01:05:00 +10:00
@@ -292,11 +310,31 @@
2019-07-29 16:36:51 +10:00
@Override
public void close() throws IOException {
- this.save(true);
+ // CraftBukkit start
+ close(true);
+ }
+
+ public void close(boolean save) throws IOException {
+ if (save) {
+ this.save(true);
+ }
+ // CraftBukkit end
this.lightEngine.close();
2021-06-11 15:00:00 +10:00
this.chunkMap.close();
2018-07-15 10:00:00 +10:00
}
2014-11-26 08:32:16 +11:00
2019-04-23 12:00:00 +10:00
+ // CraftBukkit start - modelled on below
+ public void purgeUnload() {
2021-11-22 09:00:00 +11:00
+ this.level.getProfiler().push("purge");
+ this.distanceManager.purgeStaleTickets();
+ this.runDistanceManagerUpdates();
+ this.level.getProfiler().popPush("unload");
+ this.chunkMap.tick(() -> true);
+ this.level.getProfiler().pop();
2019-05-28 06:30:00 +10:00
+ this.clearCache();
2018-08-26 12:00:00 +10:00
+ }
+ // CraftBukkit end
+
2021-06-11 15:00:00 +10:00
@Override
2022-03-01 02:00:00 +11:00
public void tick(BooleanSupplier booleansupplier, boolean flag) {
2021-11-22 09:00:00 +11:00
this.level.getProfiler().push("purge");
2024-06-14 01:05:00 +10:00
@@ -346,11 +384,11 @@
2021-11-22 09:00:00 +11:00
2023-12-06 03:40:00 +11:00
this.lastSpawnState = spawnercreature_d;
gameprofilerfiller.popPush("spawnAndTick");
- boolean flag = this.level.getGameRules().getBoolean(GameRules.RULE_DOMOBSPAWNING);
+ boolean flag = this.level.getGameRules().getBoolean(GameRules.RULE_DOMOBSPAWNING) && !this.level.players().isEmpty(); // CraftBukkit
2019-05-14 10:00:00 +10:00
2023-12-06 03:40:00 +11:00
SystemUtils.shuffle(list, this.level.random);
int l = this.level.getGameRules().getInt(GameRules.RULE_RANDOMTICKING);
- boolean flag1 = this.level.getLevelData().getGameTime() % 400L == 0L;
+ boolean flag1 = this.level.ticksPerSpawnCategory.getLong(org.bukkit.entity.SpawnCategory.ANIMAL) != 0L && this.level.getLevelData().getGameTime() % this.level.ticksPerSpawnCategory.getLong(org.bukkit.entity.SpawnCategory.ANIMAL) == 0L; // CraftBukkit
Iterator iterator1 = list.iterator();
2021-07-07 00:00:00 +10:00
2023-12-06 03:40:00 +11:00
while (iterator1.hasNext()) {
2024-06-14 01:05:00 +10:00
@@ -560,13 +598,19 @@
2021-06-11 15:00:00 +10:00
}
2019-07-13 20:19:44 +02:00
@Override
2021-11-22 09:00:00 +11:00
- protected boolean pollTask() {
2019-07-13 20:19:44 +02:00
+ // CraftBukkit start - process pending Chunk loadCallback() and unloadCallback() after each run task
2021-11-22 09:00:00 +11:00
+ public boolean pollTask() {
2019-07-13 20:19:44 +02:00
+ try {
2021-11-22 09:00:00 +11:00
if (ChunkProviderServer.this.runDistanceManagerUpdates()) {
2019-07-13 20:19:44 +02:00
return true;
} else {
2021-11-22 09:00:00 +11:00
ChunkProviderServer.this.lightEngine.tryScheduleUpdate();
return super.pollTask();
2019-07-13 20:19:44 +02:00
}
+ } finally {
2021-06-11 15:00:00 +10:00
+ chunkMap.callbackExecutor.run();
2019-07-13 20:19:44 +02:00
+ }
+ // CraftBukkit end
}
}
2021-11-22 09:00:00 +11:00