2016-03-24 23:38:38 -07:00
|
|
|
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
|
|
From: Byteflux <byte@byteflux.net>
|
|
|
|
Date: Wed, 2 Mar 2016 00:52:31 -0600
|
|
|
|
Subject: [PATCH] Lighting Queue
|
|
|
|
|
2016-07-15 17:36:53 -05:00
|
|
|
This provides option to queue lighting updates to ensure they do not cause the server lag
|
2016-03-24 23:38:38 -07:00
|
|
|
|
2016-05-15 18:48:39 -04:00
|
|
|
diff --git a/src/main/java/co/aikar/timings/WorldTimingsHandler.java b/src/main/java/co/aikar/timings/WorldTimingsHandler.java
|
2017-12-22 15:25:01 -06:00
|
|
|
index e0ad559b..4eebd9fa 100644
|
2016-05-15 18:48:39 -04:00
|
|
|
--- a/src/main/java/co/aikar/timings/WorldTimingsHandler.java
|
|
|
|
+++ b/src/main/java/co/aikar/timings/WorldTimingsHandler.java
|
|
|
|
@@ -0,0 +0,0 @@ public class WorldTimingsHandler {
|
2017-08-12 22:32:01 +01:00
|
|
|
public final Timing worldSaveLevel;
|
2016-10-21 15:42:49 -05:00
|
|
|
public final Timing chunkSaveData;
|
2016-03-24 23:38:38 -07:00
|
|
|
|
2016-05-15 18:48:39 -04:00
|
|
|
+ public final Timing lightingQueueTimer;
|
|
|
|
+
|
|
|
|
public WorldTimingsHandler(World server) {
|
|
|
|
String name = server.worldData.getName() +" - ";
|
|
|
|
|
|
|
|
@@ -0,0 +0,0 @@ public class WorldTimingsHandler {
|
|
|
|
tracker2 = Timings.ofSafe(name + "tracker stage 2");
|
|
|
|
doTick = Timings.ofSafe(name + "doTick");
|
|
|
|
tickEntities = Timings.ofSafe(name + "tickEntities");
|
|
|
|
+
|
|
|
|
+ lightingQueueTimer = Timings.ofSafe(name + "Lighting Queue");
|
|
|
|
}
|
|
|
|
}
|
2016-03-24 23:38:38 -07:00
|
|
|
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
2018-01-10 23:31:19 -06:00
|
|
|
index bf7af475..94a17304 100644
|
2016-03-24 23:38:38 -07:00
|
|
|
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
|
|
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
|
|
|
|
@@ -0,0 +0,0 @@ public class PaperWorldConfig {
|
2016-03-30 20:50:23 -04:00
|
|
|
netherVoidTopDamage = getBoolean( "nether-ceiling-void-damage", false );
|
|
|
|
log("Top of the nether void damage: " + netherVoidTopDamage);
|
2016-03-24 23:38:38 -07:00
|
|
|
}
|
2016-03-30 20:50:23 -04:00
|
|
|
+
|
2016-03-24 23:38:38 -07:00
|
|
|
+ public boolean queueLightUpdates;
|
|
|
|
+ private void queueLightUpdates() {
|
|
|
|
+ queueLightUpdates = getBoolean("queue-light-updates", false);
|
|
|
|
+ log("Lighting Queue enabled: " + queueLightUpdates);
|
|
|
|
+ }
|
|
|
|
}
|
|
|
|
diff --git a/src/main/java/net/minecraft/server/Chunk.java b/src/main/java/net/minecraft/server/Chunk.java
|
2017-12-22 15:25:01 -06:00
|
|
|
index 66c78083..c1f25e8d 100644
|
2016-03-24 23:38:38 -07:00
|
|
|
--- a/src/main/java/net/minecraft/server/Chunk.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/Chunk.java
|
|
|
|
@@ -0,0 +0,0 @@ public class Chunk {
|
2016-05-15 18:48:39 -04:00
|
|
|
private boolean m;
|
|
|
|
public final Map<BlockPosition, TileEntity> tileEntities;
|
|
|
|
public final List<Entity>[] entitySlices; // Spigot
|
|
|
|
+ final PaperLightingQueue.LightingQueue lightingQueue = new PaperLightingQueue.LightingQueue(this); // Paper
|
|
|
|
private boolean done;
|
|
|
|
private boolean lit;
|
|
|
|
private boolean r;
|
2016-03-24 23:38:38 -07:00
|
|
|
@@ -0,0 +0,0 @@ public class Chunk {
|
|
|
|
private void h(boolean flag) {
|
|
|
|
this.world.methodProfiler.a("recheckGaps");
|
|
|
|
if (this.world.areChunksLoaded(new BlockPosition(this.locX * 16 + 8, 0, this.locZ * 16 + 8), 16)) {
|
2017-12-22 15:25:01 -06:00
|
|
|
+ this.runOrQueueLightUpdate(() -> recheckGaps(flag)); // Paper - Queue light update
|
2016-03-24 23:38:38 -07:00
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ private void recheckGaps(boolean flag) {
|
|
|
|
+ if (true) {
|
|
|
|
+ // Paper end
|
|
|
|
for (int i = 0; i < 16; ++i) {
|
|
|
|
for (int j = 0; j < 16; ++j) {
|
2016-05-11 22:07:46 -04:00
|
|
|
if (this.i[i + j * 16]) {
|
2016-03-24 23:38:38 -07:00
|
|
|
@@ -0,0 +0,0 @@ public class Chunk {
|
|
|
|
if (flag) {
|
|
|
|
this.initLighting();
|
2017-12-22 15:25:01 -06:00
|
|
|
} else {
|
|
|
|
+ this.runOrQueueLightUpdate(() -> { // Paper - Queue light update
|
2016-03-24 23:38:38 -07:00
|
|
|
int j1 = iblockdata.c();
|
|
|
|
int k1 = iblockdata1.c();
|
|
|
|
|
|
|
|
@@ -0,0 +0,0 @@ public class Chunk {
|
|
|
|
if (j1 != k1 && (j1 < k1 || this.getBrightness(EnumSkyBlock.SKY, blockposition) > 0 || this.getBrightness(EnumSkyBlock.BLOCK, blockposition) > 0)) {
|
|
|
|
this.d(i, k);
|
|
|
|
}
|
2016-05-15 18:48:39 -04:00
|
|
|
+ }); // Paper
|
2016-03-24 23:38:38 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
TileEntity tileentity;
|
2017-12-22 15:25:01 -06:00
|
|
|
@@ -0,0 +0,0 @@ public class Chunk {
|
|
|
|
this.w = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
+ // Paper start
|
|
|
|
+ public void runOrQueueLightUpdate(Runnable runnable) {
|
|
|
|
+ if (this.world.paperConfig.queueLightUpdates) {
|
|
|
|
+ lightingQueue.add(runnable);
|
|
|
|
+ } else {
|
|
|
|
+ runnable.run();
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ // Paper end
|
|
|
|
+
|
|
|
|
public static enum EnumTileEntityState {
|
|
|
|
|
|
|
|
IMMEDIATE, QUEUED, CHECK;
|
2016-03-24 23:38:38 -07:00
|
|
|
diff --git a/src/main/java/net/minecraft/server/ChunkProviderServer.java b/src/main/java/net/minecraft/server/ChunkProviderServer.java
|
2017-12-22 15:25:01 -06:00
|
|
|
index 69ded6aa..e9bc23a6 100644
|
2016-03-24 23:38:38 -07:00
|
|
|
--- a/src/main/java/net/minecraft/server/ChunkProviderServer.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/ChunkProviderServer.java
|
|
|
|
@@ -0,0 +0,0 @@ public class ChunkProviderServer implements IChunkProvider {
|
2016-06-22 22:18:41 -04:00
|
|
|
return false;
|
|
|
|
}
|
2016-07-12 23:22:58 -05:00
|
|
|
save = event.isSaveChunk();
|
2016-06-22 22:18:41 -04:00
|
|
|
+ chunk.lightingQueue.processUnload(); // Paper
|
2016-03-24 23:38:38 -07:00
|
|
|
|
2016-06-22 22:18:41 -04:00
|
|
|
// Update neighbor counts
|
|
|
|
for (int x = -2; x < 3; x++) {
|
2016-03-24 23:38:38 -07:00
|
|
|
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
|
2017-12-22 15:25:01 -06:00
|
|
|
index 47fe9262..d63a243f 100644
|
2016-03-24 23:38:38 -07:00
|
|
|
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
|
2017-05-14 13:05:01 -05:00
|
|
|
@@ -0,0 +0,0 @@ public abstract class MinecraftServer implements ICommandListener, Runnable, IAs
|
2016-05-15 18:48:39 -04:00
|
|
|
protected void C() throws ExceptionWorldConflict { // CraftBukkit - added throws
|
|
|
|
co.aikar.timings.TimingsManager.FULL_SERVER_TICK.startTiming(); // Paper
|
2017-01-30 22:33:54 -06:00
|
|
|
this.slackActivityAccountant.tickStarted(); // Spigot
|
2016-05-15 18:48:39 -04:00
|
|
|
- long i = System.nanoTime();
|
|
|
|
+ long i = System.nanoTime(); long startTime = i; // Paper
|
|
|
|
|
|
|
|
++this.ticks;
|
2016-11-16 20:23:38 -06:00
|
|
|
if (this.T) {
|
2017-05-14 13:05:01 -05:00
|
|
|
@@ -0,0 +0,0 @@ public abstract class MinecraftServer implements ICommandListener, Runnable, IAs
|
2016-03-24 23:38:38 -07:00
|
|
|
this.methodProfiler.b();
|
2016-04-29 21:23:40 -04:00
|
|
|
|
2016-05-15 18:48:39 -04:00
|
|
|
org.spigotmc.WatchdogThread.tick(); // Spigot
|
|
|
|
+ PaperLightingQueue.processQueue(startTime); // Paper
|
2017-01-30 22:33:54 -06:00
|
|
|
this.slackActivityAccountant.tickEnded(tickNanos); // Spigot
|
2016-05-15 18:48:39 -04:00
|
|
|
co.aikar.timings.TimingsManager.FULL_SERVER_TICK.stopTiming(); // Paper
|
|
|
|
}
|
|
|
|
diff --git a/src/main/java/net/minecraft/server/PaperLightingQueue.java b/src/main/java/net/minecraft/server/PaperLightingQueue.java
|
|
|
|
new file mode 100644
|
2017-12-22 15:25:01 -06:00
|
|
|
index 00000000..345cd582
|
2016-05-15 18:48:39 -04:00
|
|
|
--- /dev/null
|
|
|
|
+++ b/src/main/java/net/minecraft/server/PaperLightingQueue.java
|
|
|
|
@@ -0,0 +0,0 @@
|
|
|
|
+package net.minecraft.server;
|
|
|
|
+
|
|
|
|
+import co.aikar.timings.Timing;
|
2017-01-02 14:08:55 -05:00
|
|
|
+import it.unimi.dsi.fastutil.objects.ObjectCollection;
|
|
|
|
+
|
2016-05-15 18:48:39 -04:00
|
|
|
+import java.util.ArrayDeque;
|
|
|
|
+
|
|
|
|
+class PaperLightingQueue {
|
|
|
|
+ private static final long MAX_TIME = (long) (1000000000 / 20 * .95);
|
|
|
|
+ private static int updatesThisTick;
|
|
|
|
+
|
|
|
|
+
|
|
|
|
+ static void processQueue(long curTime) {
|
|
|
|
+ updatesThisTick = 0;
|
2016-03-24 23:38:38 -07:00
|
|
|
+
|
2016-05-15 18:48:39 -04:00
|
|
|
+ final long startTime = System.nanoTime();
|
|
|
|
+ final long maxTickTime = MAX_TIME - (startTime - curTime);
|
|
|
|
+
|
|
|
|
+ START:
|
|
|
|
+ for (World world : MinecraftServer.getServer().worlds) {
|
|
|
|
+ if (!world.paperConfig.queueLightUpdates) {
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+
|
2017-01-02 14:08:55 -05:00
|
|
|
+ ObjectCollection<Chunk> loadedChunks = ((WorldServer) world).getChunkProviderServer().chunks.values();
|
|
|
|
+ for (Chunk chunk : loadedChunks.toArray(new Chunk[loadedChunks.size()])) {
|
2016-05-15 18:48:39 -04:00
|
|
|
+ if (chunk.lightingQueue.processQueue(startTime, maxTickTime)) {
|
|
|
|
+ break START;
|
2016-03-24 23:38:38 -07:00
|
|
|
+ }
|
|
|
|
+ }
|
2016-05-15 18:48:39 -04:00
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ static class LightingQueue extends ArrayDeque<Runnable> {
|
|
|
|
+ final private Chunk chunk;
|
2016-03-24 23:38:38 -07:00
|
|
|
+
|
2016-05-15 18:48:39 -04:00
|
|
|
+ LightingQueue(Chunk chunk) {
|
|
|
|
+ super();
|
|
|
|
+ this.chunk = chunk;
|
2016-03-24 23:38:38 -07:00
|
|
|
+ }
|
|
|
|
+
|
2016-05-15 18:48:39 -04:00
|
|
|
+ /**
|
|
|
|
+ * Processes the lighting queue for this chunk
|
|
|
|
+ *
|
|
|
|
+ * @param startTime If start Time is 0, we will not limit execution time
|
|
|
|
+ * @param maxTickTime Maximum time to spend processing lighting updates
|
|
|
|
+ * @return true to abort processing furthur lighting updates
|
|
|
|
+ */
|
|
|
|
+ private boolean processQueue(long startTime, long maxTickTime) {
|
|
|
|
+ if (this.isEmpty()) {
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+ try (Timing ignored = chunk.world.timings.lightingQueueTimer.startTiming()) {
|
|
|
|
+ Runnable lightUpdate;
|
|
|
|
+ while ((lightUpdate = this.poll()) != null) {
|
|
|
|
+ lightUpdate.run();
|
|
|
|
+ if (startTime > 0 && ++PaperLightingQueue.updatesThisTick % 10 == 0 && PaperLightingQueue.updatesThisTick > 10) {
|
|
|
|
+ if (System.nanoTime() - startTime > maxTickTime) {
|
|
|
|
+ return true;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ return false;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * Flushes lighting updates to unload the chunk
|
|
|
|
+ */
|
|
|
|
+ void processUnload() {
|
|
|
|
+ if (!chunk.world.paperConfig.queueLightUpdates) {
|
|
|
|
+ return;
|
|
|
|
+ }
|
|
|
|
+ processQueue(0, 0); // No timeout
|
|
|
|
+
|
|
|
|
+ final int radius = 1; // TODO: bitflip, why should this ever be 2?
|
|
|
|
+ for (int x = chunk.locX - radius; x <= chunk.locX + radius; ++x) {
|
|
|
|
+ for (int z = chunk.locZ - radius; z <= chunk.locZ + radius; ++z) {
|
|
|
|
+ if (x == chunk.locX && z == chunk.locZ) {
|
|
|
|
+ continue;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ Chunk neighbor = MCUtil.getLoadedChunkWithoutMarkingActive(chunk.world, x, z);
|
|
|
|
+ if (neighbor != null) {
|
|
|
|
+ neighbor.lightingQueue.processQueue(0, 0); // No timeout
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+}
|
2016-03-24 23:38:38 -07:00
|
|
|
diff --git a/src/main/java/net/minecraft/server/World.java b/src/main/java/net/minecraft/server/World.java
|
2017-12-22 15:25:01 -06:00
|
|
|
index fd5f8102..77ed2d24 100644
|
2016-03-24 23:38:38 -07:00
|
|
|
--- a/src/main/java/net/minecraft/server/World.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/World.java
|
|
|
|
@@ -0,0 +0,0 @@ public abstract class World implements IBlockAccess {
|
|
|
|
} else {
|
|
|
|
if (iblockdata.c() != iblockdata1.c() || iblockdata.d() != iblockdata1.d()) {
|
|
|
|
this.methodProfiler.a("checkLight");
|
|
|
|
- this.w(blockposition);
|
2017-12-22 15:25:01 -06:00
|
|
|
+ chunk.runOrQueueLightUpdate(() -> this.w(blockposition)); // Paper - Queue light update
|
2016-03-24 23:38:38 -07:00
|
|
|
this.methodProfiler.b();
|
|
|
|
}
|
|
|
|
|
|
|
|
--
|