mirror of
https://github.com/Minestom/Minestom.git
synced 2025-01-17 21:51:33 +01:00
Use soft references for pooled buffers
Signed-off-by: TheMode <themode@outlook.fr>
This commit is contained in:
parent
538ef75552
commit
df92939ddd
@ -27,6 +27,7 @@ import javax.crypto.Cipher;
|
||||
import javax.crypto.SecretKey;
|
||||
import javax.crypto.ShortBufferException;
|
||||
import java.io.IOException;
|
||||
import java.lang.ref.SoftReference;
|
||||
import java.net.SocketAddress;
|
||||
import java.nio.BufferUnderflowException;
|
||||
import java.nio.ByteBuffer;
|
||||
@ -44,7 +45,7 @@ import java.util.zip.DataFormatException;
|
||||
@ApiStatus.Internal
|
||||
public class PlayerSocketConnection extends PlayerConnection {
|
||||
private final static Logger LOGGER = LoggerFactory.getLogger(PlayerSocketConnection.class);
|
||||
private final static Queue<BinaryBuffer> POOLED_BUFFERS = new ConcurrentLinkedQueue<>();
|
||||
private final static Queue<SoftReference<BinaryBuffer>> POOLED_BUFFERS = new ConcurrentLinkedQueue<>();
|
||||
private final static int BUFFER_SIZE = 262_143;
|
||||
|
||||
private final Worker worker;
|
||||
@ -256,26 +257,29 @@ public class PlayerSocketConnection extends PlayerConnection {
|
||||
|
||||
@Override
|
||||
public void flush() {
|
||||
boolean shouldDisconnect = false;
|
||||
if (!channel.isOpen()) return;
|
||||
if (tickBuffer.readableBytes() == 0 && waitingBuffers.isEmpty()) return;
|
||||
synchronized (bufferLock) {
|
||||
if (tickBuffer.readableBytes() == 0 && waitingBuffers.isEmpty()) return;
|
||||
final BinaryBuffer localBuffer = this.tickBuffer;
|
||||
if (localBuffer.readableBytes() == 0 && waitingBuffers.isEmpty()) return;
|
||||
// Update tick buffer
|
||||
this.tickBuffer = getPooledBuffer();
|
||||
if (encrypted) {
|
||||
final Cipher cipher = encryptCipher;
|
||||
// Encrypt data first
|
||||
final int remainingBytes = tickBuffer.readableBytes();
|
||||
final byte[] bytes = tickBuffer.readRemainingBytes();
|
||||
final int remainingBytes = localBuffer.readableBytes();
|
||||
final byte[] bytes = localBuffer.readRemainingBytes();
|
||||
byte[] outTempArray = new byte[cipher.getOutputSize(remainingBytes)];
|
||||
try {
|
||||
cipher.update(bytes, 0, remainingBytes, outTempArray);
|
||||
} catch (ShortBufferException e) {
|
||||
MinecraftServer.getExceptionManager().handleException(e);
|
||||
}
|
||||
this.tickBuffer.clear();
|
||||
this.tickBuffer.writeBytes(outTempArray);
|
||||
localBuffer.clear();
|
||||
localBuffer.writeBytes(outTempArray);
|
||||
}
|
||||
|
||||
this.waitingBuffers.add(tickBuffer);
|
||||
this.waitingBuffers.add(localBuffer);
|
||||
Iterator<BinaryBuffer> iterator = waitingBuffers.iterator();
|
||||
while (iterator.hasNext()) {
|
||||
BinaryBuffer waitingBuffer = iterator.next();
|
||||
@ -283,14 +287,20 @@ public class PlayerSocketConnection extends PlayerConnection {
|
||||
if (!waitingBuffer.writeChannel(channel)) break;
|
||||
iterator.remove();
|
||||
waitingBuffer.clear();
|
||||
POOLED_BUFFERS.add(waitingBuffer);
|
||||
POOLED_BUFFERS.add(new SoftReference<>(waitingBuffer));
|
||||
} catch (IOException e) {
|
||||
final String message = e.getMessage();
|
||||
if (message == null ||
|
||||
(!message.equals("Broken pipe") && !message.equals("Connection reset by peer"))) {
|
||||
MinecraftServer.getExceptionManager().handleException(e);
|
||||
}
|
||||
shouldDisconnect = true;
|
||||
}
|
||||
}
|
||||
// Update tick buffer
|
||||
this.tickBuffer = getPooledBuffer();
|
||||
}
|
||||
if (shouldDisconnect) disconnect();
|
||||
}
|
||||
|
||||
@Override
|
||||
@ -314,7 +324,10 @@ public class PlayerSocketConnection extends PlayerConnection {
|
||||
public void disconnect() {
|
||||
this.worker.disconnect(this, channel);
|
||||
synchronized (bufferLock) {
|
||||
POOLED_BUFFERS.addAll(waitingBuffers);
|
||||
for (BinaryBuffer waitingBuffer : waitingBuffers) {
|
||||
POOLED_BUFFERS.add(new SoftReference<>(waitingBuffer));
|
||||
}
|
||||
this.waitingBuffers.clear();
|
||||
}
|
||||
}
|
||||
|
||||
@ -452,12 +465,17 @@ public class PlayerSocketConnection extends PlayerConnection {
|
||||
}
|
||||
|
||||
private static BinaryBuffer getPooledBuffer() {
|
||||
BinaryBuffer newBuffer = POOLED_BUFFERS.poll();
|
||||
if (newBuffer == null) {
|
||||
newBuffer = BinaryBuffer.ofSize(BUFFER_SIZE);
|
||||
BinaryBuffer buffer = null;
|
||||
SoftReference<BinaryBuffer> ref;
|
||||
while ((ref = POOLED_BUFFERS.poll()) != null) {
|
||||
buffer = ref.get();
|
||||
if (buffer != null) break;
|
||||
}
|
||||
if (buffer == null) {
|
||||
buffer = BinaryBuffer.ofSize(BUFFER_SIZE);
|
||||
} else {
|
||||
newBuffer.clear();
|
||||
buffer.clear();
|
||||
}
|
||||
return newBuffer;
|
||||
return buffer;
|
||||
}
|
||||
}
|
||||
|
Loading…
Reference in New Issue
Block a user