2020-04-03 05:07:06 +08:00
From f257bfd57e675b1d5c23dc98d9f0e7afb4cf0745 Mon Sep 17 00:00:00 2001
2016-01-13 13:00:58 +08:00
From: Aikar <aikar@aikar.co>
2016-03-01 07:09:49 +08:00
Date: Thu, 3 Mar 2016 01:17:12 -0600
2016-01-13 13:00:58 +08:00
Subject: [PATCH] Ensure commands are not ran async
Plugins calling Player.chat("/foo") or Server.dispatchCommand() could
trigger the server to execute a command while on another thread.
These commands would then process EXPECTING to be on the main thread, leaving to
very hard to trace concurrency issues.
This change will synchronize the command execution back to the main thread, causing a
big slowdown in execution but throwing an exception at same time to raise awareness
that it is happening so that plugin authors can fix their code to stop executing commands async.
diff --git a/src/main/java/net/minecraft/server/PlayerConnection.java b/src/main/java/net/minecraft/server/PlayerConnection.java
2020-04-03 05:07:06 +08:00
index 914366afcd..5e8739462e 100644
2016-01-13 13:00:58 +08:00
--- a/src/main/java/net/minecraft/server/PlayerConnection.java
+++ b/src/main/java/net/minecraft/server/PlayerConnection.java
2020-02-09 08:32:48 +08:00
@@ -1570,6 +1570,29 @@ public class PlayerConnection implements PacketListenerPlayIn {
2016-01-13 13:00:58 +08:00
}
if (!async && s.startsWith("/")) {
2016-03-01 07:09:49 +08:00
+ // Paper Start
2018-02-15 13:34:58 +08:00
+ if (!org.spigotmc.AsyncCatcher.shuttingDown && !org.bukkit.Bukkit.isPrimaryThread()) {
2016-01-13 13:00:58 +08:00
+ final String fCommandLine = s;
+ MinecraftServer.LOGGER.log(org.apache.logging.log4j.Level.ERROR, "Command Dispatched Async: " + fCommandLine);
+ MinecraftServer.LOGGER.log(org.apache.logging.log4j.Level.ERROR, "Please notify author of plugin causing this execution to fix this bug! see: http://bit.ly/1oSiM6C", new Throwable());
+ Waitable wait = new Waitable() {
+ @Override
+ protected Object evaluate() {
+ chat(fCommandLine, false);
+ return null;
+ }
+ };
+ minecraftServer.processQueue.add(wait);
+ try {
+ wait.get();
+ return;
+ } catch (InterruptedException e) {
+ Thread.currentThread().interrupt(); // This is proper habit for java. If we aren't handling it, pass it on!
+ } catch (Exception e) {
+ throw new RuntimeException("Exception processing chat command", e.getCause());
+ }
+ }
2016-03-01 07:09:49 +08:00
+ // Paper End
2016-01-13 13:00:58 +08:00
this.handleCommand(s);
2019-04-25 14:53:51 +08:00
} else if (this.player.getChatFlags() == EnumChatVisibility.SYSTEM) {
2016-01-13 13:00:58 +08:00
// Do nothing, this is coming from a plugin
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
2020-04-03 05:07:06 +08:00
index e3939387e7..f7c29e372f 100644
2016-01-13 13:00:58 +08:00
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
2020-03-10 19:01:15 +08:00
@@ -720,6 +720,29 @@ public final class CraftServer implements Server {
2016-01-13 13:00:58 +08:00
Validate.notNull(commandLine, "CommandLine cannot be null");
2019-07-24 03:17:32 +08:00
org.spigotmc.AsyncCatcher.catchOp("command dispatch"); // Spigot
2016-01-13 13:00:58 +08:00
2016-03-01 07:09:49 +08:00
+ // Paper Start
2018-02-15 13:34:58 +08:00
+ if (!org.spigotmc.AsyncCatcher.shuttingDown && !Bukkit.isPrimaryThread()) {
2016-01-13 13:00:58 +08:00
+ final CommandSender fSender = sender;
+ final String fCommandLine = commandLine;
+ Bukkit.getLogger().log(Level.SEVERE, "Command Dispatched Async: " + commandLine);
+ Bukkit.getLogger().log(Level.SEVERE, "Please notify author of plugin causing this execution to fix this bug! see: http://bit.ly/1oSiM6C", new Throwable());
+ org.bukkit.craftbukkit.util.Waitable<Boolean> wait = new org.bukkit.craftbukkit.util.Waitable<Boolean>() {
+ @Override
+ protected Boolean evaluate() {
+ return dispatchCommand(fSender, fCommandLine);
+ }
+ };
+ net.minecraft.server.MinecraftServer.getServer().processQueue.add(wait);
+ try {
+ return wait.get();
+ } catch (InterruptedException e) {
+ Thread.currentThread().interrupt(); // This is proper habit for java. If we aren't handling it, pass it on!
+ } catch (Exception e) {
+ throw new RuntimeException("Exception processing dispatch command", e.getCause());
+ }
+ }
2016-03-01 07:09:49 +08:00
+ // Paper End
2016-01-13 13:00:58 +08:00
+
if (commandMap.dispatch(sender, commandLine)) {
return true;
}
2018-02-15 13:34:58 +08:00
diff --git a/src/main/java/org/bukkit/craftbukkit/util/ServerShutdownThread.java b/src/main/java/org/bukkit/craftbukkit/util/ServerShutdownThread.java
2020-04-03 05:07:06 +08:00
index ddef523ea8..70f8d42992 100644
2018-02-15 13:34:58 +08:00
--- a/src/main/java/org/bukkit/craftbukkit/util/ServerShutdownThread.java
+++ b/src/main/java/org/bukkit/craftbukkit/util/ServerShutdownThread.java
2019-04-25 14:53:51 +08:00
@@ -13,6 +13,7 @@ public class ServerShutdownThread extends Thread {
2018-02-15 13:34:58 +08:00
public void run() {
try {
org.spigotmc.AsyncCatcher.enabled = false; // Spigot
+ org.spigotmc.AsyncCatcher.shuttingDown = true; // Paper
2019-04-25 14:53:51 +08:00
server.close();
} finally {
try {
2018-02-15 13:34:58 +08:00
diff --git a/src/main/java/org/spigotmc/AsyncCatcher.java b/src/main/java/org/spigotmc/AsyncCatcher.java
2020-04-03 05:07:06 +08:00
index aeed769725..9f7d2ef932 100644
2018-02-15 13:34:58 +08:00
--- a/src/main/java/org/spigotmc/AsyncCatcher.java
+++ b/src/main/java/org/spigotmc/AsyncCatcher.java
@@ -6,6 +6,7 @@ public class AsyncCatcher
{
public static boolean enabled = true;
+ public static boolean shuttingDown = false; // Paper
public static void catchOp(String reason)
{
diff --git a/src/main/java/org/spigotmc/RestartCommand.java b/src/main/java/org/spigotmc/RestartCommand.java
2020-04-03 05:07:06 +08:00
index e7b953ca31..ccea803f58 100644
2018-02-15 13:34:58 +08:00
--- a/src/main/java/org/spigotmc/RestartCommand.java
+++ b/src/main/java/org/spigotmc/RestartCommand.java
@@ -43,6 +43,7 @@ public class RestartCommand extends Command
2019-01-01 11:15:55 +08:00
private static void restart(final String restartScript)
2018-02-15 13:34:58 +08:00
{
AsyncCatcher.enabled = false; // Disable async catcher incase it interferes with us
+ org.spigotmc.AsyncCatcher.shuttingDown = true; // Paper
try
{
2019-01-01 11:15:55 +08:00
String[] split = restartScript.split( " " );
2016-01-13 13:00:58 +08:00
--
2020-04-03 05:07:06 +08:00
2.25.1
2016-01-13 13:00:58 +08:00