From ede60970b933ff6e5c2070abceae9b1df3bb42e6 Mon Sep 17 00:00:00 2001 From: "Kristian S. Stangeland" Date: Tue, 4 Dec 2012 16:21:37 +0100 Subject: [PATCH] Ensure that we don't cache wrong methods in the serialization. --- ProtocolLib/dependency-reduced-pom.xml | 2 +- .../protocol/events/PacketContainer.java | 60 ++++++++++++++----- .../injector/player/InjectedArrayList.java | 1 - 3 files changed, 46 insertions(+), 17 deletions(-) diff --git a/ProtocolLib/dependency-reduced-pom.xml b/ProtocolLib/dependency-reduced-pom.xml index 8bf4ad99..7f3c8e0c 100644 --- a/ProtocolLib/dependency-reduced-pom.xml +++ b/ProtocolLib/dependency-reduced-pom.xml @@ -4,7 +4,7 @@ com.comphenix.protocol ProtocolLib ProtocolLib - 1.7.1 + 1.7.2-SNAPSHOT Provides read/write access to the Minecraft protocol. http://dev.bukkit.org/server-mods/protocollib/ diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java b/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java index 2ff67143..6f424874 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/events/PacketContainer.java @@ -29,6 +29,7 @@ import java.lang.reflect.InvocationTargetException; import java.lang.reflect.Method; import java.util.Collection; import java.util.List; +import java.util.concurrent.ConcurrentMap; import org.bukkit.World; import org.bukkit.WorldType; @@ -43,6 +44,7 @@ import com.comphenix.protocol.wrappers.BukkitConverters; import com.comphenix.protocol.wrappers.ChunkPosition; import com.comphenix.protocol.wrappers.WrappedDataWatcher; import com.comphenix.protocol.wrappers.WrappedWatchableObject; +import com.google.common.collect.Maps; import net.minecraft.server.Packet; @@ -63,11 +65,11 @@ public class PacketContainer implements Serializable { // Current structure modifier protected transient StructureModifier structureModifier; - + // Support for serialization - private static Method writeMethod; - private static Method readMethod; - + private static ConcurrentMap, Method> writeMethods = Maps.newConcurrentMap(); + private static ConcurrentMap, Method> readMethods = Maps.newConcurrentMap(); + /** * Creates a packet container for a new packet. * @param id - ID of the packet to create. @@ -100,6 +102,12 @@ public class PacketContainer implements Serializable { this.structureModifier = structure; } + /** + * For serialization. + */ + protected PacketContainer() { + } + /** * Retrieves the underlying Minecraft packet. * @return Underlying Minecraft packet. @@ -399,14 +407,12 @@ public class PacketContainer implements Serializable { // We'll take care of NULL packets as well output.writeBoolean(handle != null); - - // Retrieve the write method by reflection - if (writeMethod == null) - writeMethod = FuzzyReflection.fromObject(handle).getMethodByParameters("write", DataOutputStream.class); - + try { // Call the write-method - writeMethod.invoke(handle, new DataOutputStream(output)); + getMethodLazily(writeMethods, handle.getClass(), "write", DataOutputStream.class). + invoke(handle, new DataOutputStream(output)); + } catch (IllegalArgumentException e) { throw new IOException("Minecraft packet doesn't support DataOutputStream", e); } catch (IllegalAccessException e) { @@ -429,13 +435,11 @@ public class PacketContainer implements Serializable { // Create a default instance of the packet handle = StructureCache.newPacket(id); - // Retrieve the read method by reflection - if (readMethod == null) - readMethod = FuzzyReflection.fromObject(handle).getMethodByParameters("read", DataInputStream.class); - // Call the read method try { - readMethod.invoke(handle, new DataInputStream(input)); + getMethodLazily(readMethods, handle.getClass(), "read", DataInputStream.class). + invoke(handle, new DataInputStream(input)); + } catch (IllegalArgumentException e) { throw new IOException("Minecraft packet doesn't support DataInputStream", e); } catch (IllegalAccessException e) { @@ -448,4 +452,30 @@ public class PacketContainer implements Serializable { structureModifier = structureModifier.withTarget(handle); } } + + /** + * Retrieve the cached method concurrently. + * @param lookup - a lazy lookup cache. + * @param handleClass - class type of the current packet. + * @param methodName - name of method to retrieve. + * @param parameterClass - the one parameter type in the method. + * @return Reflected method. + */ + private Method getMethodLazily(ConcurrentMap, Method> lookup, + Class handleClass, String methodName, Class parameterClass) { + Method method = lookup.get(handleClass); + + // Atomic operation + if (method == null) { + Method initialized = FuzzyReflection.fromClass(handleClass).getMethodByParameters(methodName, parameterClass); + method = lookup.putIfAbsent(handleClass, initialized); + + // Use our version if we succeeded + if (method == null) { + method = initialized; + } + } + + return method; + } } diff --git a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedArrayList.java b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedArrayList.java index d92d4803..2f8ee829 100644 --- a/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedArrayList.java +++ b/ProtocolLib/src/main/java/com/comphenix/protocol/injector/player/InjectedArrayList.java @@ -27,7 +27,6 @@ import com.comphenix.protocol.injector.player.NetworkFieldInjector.FakePacket; import net.minecraft.server.Packet; import net.sf.cglib.proxy.Enhancer; -import net.sf.cglib.proxy.Factory; import net.sf.cglib.proxy.MethodInterceptor; import net.sf.cglib.proxy.MethodProxy;