mirror of
https://github.com/Minestom/Minestom.git
synced 2025-01-23 16:41:35 +01:00
WIP BinaryBuffer
This commit is contained in:
parent
aa2a6522dc
commit
455c21208e
@ -127,9 +127,7 @@ public final class PacketProcessor {
|
|||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
final Player player = connection.getPlayer();
|
final Player player = connection.getPlayer();
|
||||||
final String username = player != null ? player.getUsername() : "null";
|
final String username = player != null ? player.getUsername() : "null";
|
||||||
LOGGER.warn("Connection {} ({}) sent an unexpected packet.",
|
LOGGER.warn("Connection {} ({}) sent an unexpected packet.", connection.getRemoteAddress(), username);
|
||||||
connection.getRemoteAddress(),
|
|
||||||
username);
|
|
||||||
MinecraftServer.getExceptionManager().handleException(e);
|
MinecraftServer.getExceptionManager().handleException(e);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -14,7 +14,7 @@ import net.minestom.server.network.packet.server.login.SetCompressionPacket;
|
|||||||
import net.minestom.server.network.socket.Server;
|
import net.minestom.server.network.socket.Server;
|
||||||
import net.minestom.server.network.socket.Worker;
|
import net.minestom.server.network.socket.Worker;
|
||||||
import net.minestom.server.utils.PacketUtils;
|
import net.minestom.server.utils.PacketUtils;
|
||||||
import net.minestom.server.utils.Utils;
|
import net.minestom.server.utils.binary.BinaryBuffer;
|
||||||
import net.minestom.server.utils.validate.Check;
|
import net.minestom.server.utils.validate.Check;
|
||||||
import org.jetbrains.annotations.ApiStatus;
|
import org.jetbrains.annotations.ApiStatus;
|
||||||
import org.jetbrains.annotations.NotNull;
|
import org.jetbrains.annotations.NotNull;
|
||||||
@ -62,8 +62,8 @@ public class NettyPlayerConnection extends PlayerConnection {
|
|||||||
private UUID bungeeUuid;
|
private UUID bungeeUuid;
|
||||||
private PlayerSkin bungeeSkin;
|
private PlayerSkin bungeeSkin;
|
||||||
|
|
||||||
private final ByteBuffer tickBuffer = ByteBuffer.allocateDirect(Server.SOCKET_BUFFER_SIZE);
|
private final BinaryBuffer tickBuffer = BinaryBuffer.ofSize(Server.SOCKET_BUFFER_SIZE);
|
||||||
private volatile ByteBuffer cacheBuffer;
|
private volatile BinaryBuffer cacheBuffer;
|
||||||
|
|
||||||
public NettyPlayerConnection(@NotNull Worker worker, @NotNull SocketChannel channel, SocketAddress remoteAddress) {
|
public NettyPlayerConnection(@NotNull Worker worker, @NotNull SocketChannel channel, SocketAddress remoteAddress) {
|
||||||
super();
|
super();
|
||||||
@ -74,74 +74,69 @@ public class NettyPlayerConnection extends PlayerConnection {
|
|||||||
|
|
||||||
public void processPackets(Worker.Context workerContext, PacketProcessor packetProcessor) {
|
public void processPackets(Worker.Context workerContext, PacketProcessor packetProcessor) {
|
||||||
final var readBuffer = workerContext.readBuffer;
|
final var readBuffer = workerContext.readBuffer;
|
||||||
final int limit = readBuffer.limit();
|
final int limit = readBuffer.writerOffset();
|
||||||
// Read all packets
|
// Read all packets
|
||||||
while (readBuffer.remaining() > 0) {
|
while (readBuffer.readableBytes() > 0) {
|
||||||
readBuffer.mark(); // Mark the beginning of the packet
|
final var beginMark = readBuffer.mark();
|
||||||
try {
|
try {
|
||||||
// Read packet
|
// Ensure that the buffer contains the full packet (or wait for next socket read)
|
||||||
final int packetLength = Utils.readVarInt(readBuffer);
|
final int packetLength = readBuffer.readVarInt();
|
||||||
final int packetEnd = readBuffer.position() + packetLength;
|
final int packetEnd = readBuffer.readerOffset() + packetLength;
|
||||||
if (packetEnd > readBuffer.limit()) {
|
if (packetEnd > readBuffer.writerOffset()) {
|
||||||
// Integrity fail
|
// Integrity fail
|
||||||
throw new BufferUnderflowException();
|
throw new BufferUnderflowException();
|
||||||
}
|
}
|
||||||
|
// Read packet https://wiki.vg/Protocol#Packet_format
|
||||||
readBuffer.limit(packetEnd); // Ensure that the reader doesn't exceed packet bound
|
BinaryBuffer content;
|
||||||
|
|
||||||
// Read protocol
|
|
||||||
ByteBuffer content;
|
|
||||||
if (!compressed) {
|
if (!compressed) {
|
||||||
// Compression disabled, payload is following
|
// Compression disabled, payload is following
|
||||||
content = readBuffer;
|
content = readBuffer;
|
||||||
} else {
|
} else {
|
||||||
final int dataLength = Utils.readVarInt(readBuffer);
|
final int dataLength = readBuffer.readVarInt();
|
||||||
if (dataLength == 0) {
|
if (dataLength == 0) {
|
||||||
// Data is too small to be compressed, payload is following
|
// Data is too small to be compressed, payload is following
|
||||||
content = readBuffer;
|
content = readBuffer;
|
||||||
} else {
|
} else {
|
||||||
// Decompress to content buffer
|
// Decompress to content buffer
|
||||||
content = workerContext.contentBuffer;
|
content = workerContext.contentBuffer;
|
||||||
|
final var contentStartMark = content.mark();
|
||||||
try {
|
try {
|
||||||
final var inflater = workerContext.inflater;
|
final var inflater = workerContext.inflater;
|
||||||
inflater.setInput(readBuffer);
|
inflater.setInput(readBuffer.asByteBuffer(readBuffer.readerOffset(), packetEnd));
|
||||||
inflater.inflate(content);
|
inflater.inflate(content.asByteBuffer(0, content.capacity()));
|
||||||
inflater.reset();
|
inflater.reset();
|
||||||
} catch (DataFormatException e) {
|
} catch (DataFormatException e) {
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
}
|
}
|
||||||
content.flip();
|
content.reset(contentStartMark);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Process packet
|
// Process packet
|
||||||
final int packetId = Utils.readVarInt(content);
|
final int packetId = content.readVarInt();
|
||||||
try {
|
try {
|
||||||
packetProcessor.process(this, packetId, content);
|
var finalBuffer = content.asByteBuffer(content.readerOffset(), packetEnd);
|
||||||
|
packetProcessor.process(this, packetId, finalBuffer);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
// Error while reading the packet
|
// Error while reading the packet
|
||||||
MinecraftServer.getExceptionManager().handleException(e);
|
MinecraftServer.getExceptionManager().handleException(e);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return to original state (before writing)
|
// Return to original state (before writing)
|
||||||
readBuffer.limit(limit).position(packetEnd);
|
readBuffer.reset(packetEnd, limit);
|
||||||
} catch (BufferUnderflowException e) {
|
} catch (BufferUnderflowException e) {
|
||||||
readBuffer.reset();
|
readBuffer.reset(beginMark);
|
||||||
this.cacheBuffer = ByteBuffer.allocateDirect(readBuffer.remaining())
|
this.cacheBuffer = BinaryBuffer.copy(readBuffer);
|
||||||
.put(readBuffer).flip();
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void consumeCache(ByteBuffer buffer) {
|
public void consumeCache(BinaryBuffer buffer) {
|
||||||
if (cacheBuffer == null) {
|
if (cacheBuffer != null) {
|
||||||
return;
|
buffer.write(cacheBuffer);
|
||||||
}
|
|
||||||
buffer.put(cacheBuffer);
|
|
||||||
this.cacheBuffer = null;
|
this.cacheBuffer = null;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets the encryption key and add the codecs to the pipeline.
|
* Sets the encryption key and add the codecs to the pipeline.
|
||||||
@ -197,11 +192,11 @@ public class NettyPlayerConnection extends PlayerConnection {
|
|||||||
public void write(@NotNull ByteBuffer buffer) {
|
public void write(@NotNull ByteBuffer buffer) {
|
||||||
synchronized (tickBuffer) {
|
synchronized (tickBuffer) {
|
||||||
buffer.flip();
|
buffer.flip();
|
||||||
if (buffer.limit() > tickBuffer.remaining()) {
|
if (!tickBuffer.canWrite(buffer.limit())) {
|
||||||
// Tick buffer is full, flush before appending
|
// Tick buffer is full, flush before appending
|
||||||
flush();
|
flush();
|
||||||
}
|
}
|
||||||
this.tickBuffer.put(buffer);
|
this.tickBuffer.write(buffer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -224,9 +219,9 @@ public class NettyPlayerConnection extends PlayerConnection {
|
|||||||
public void flush() {
|
public void flush() {
|
||||||
if (!channel.isOpen()) return;
|
if (!channel.isOpen()) return;
|
||||||
synchronized (tickBuffer) {
|
synchronized (tickBuffer) {
|
||||||
if (tickBuffer.position() == 0) return;
|
if (tickBuffer.readableBytes() == 0) return;
|
||||||
try {
|
try {
|
||||||
this.channel.write(tickBuffer.flip());
|
this.tickBuffer.writeChannel(channel);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
MinecraftServer.getExceptionManager().handleException(e);
|
MinecraftServer.getExceptionManager().handleException(e);
|
||||||
} finally {
|
} finally {
|
||||||
|
@ -4,10 +4,10 @@ import net.minestom.server.MinecraftServer;
|
|||||||
import net.minestom.server.entity.Player;
|
import net.minestom.server.entity.Player;
|
||||||
import net.minestom.server.network.PacketProcessor;
|
import net.minestom.server.network.PacketProcessor;
|
||||||
import net.minestom.server.network.player.NettyPlayerConnection;
|
import net.minestom.server.network.player.NettyPlayerConnection;
|
||||||
|
import net.minestom.server.utils.binary.BinaryBuffer;
|
||||||
import org.jetbrains.annotations.ApiStatus;
|
import org.jetbrains.annotations.ApiStatus;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.nio.ByteBuffer;
|
|
||||||
import java.nio.channels.SelectionKey;
|
import java.nio.channels.SelectionKey;
|
||||||
import java.nio.channels.Selector;
|
import java.nio.channels.Selector;
|
||||||
import java.nio.channels.SocketChannel;
|
import java.nio.channels.SocketChannel;
|
||||||
@ -48,16 +48,11 @@ public final class Worker {
|
|||||||
}
|
}
|
||||||
var connection = connectionMap.get(channel);
|
var connection = connectionMap.get(channel);
|
||||||
try {
|
try {
|
||||||
ByteBuffer readBuffer = workerContext.readBuffer;
|
var readBuffer = workerContext.readBuffer;
|
||||||
// Consume last incomplete packet
|
// Consume last incomplete packet
|
||||||
connection.consumeCache(readBuffer);
|
connection.consumeCache(readBuffer);
|
||||||
// Read socket
|
// Read & process
|
||||||
if (channel.read(readBuffer) == -1) {
|
readBuffer.readChannel(channel);
|
||||||
// EOS
|
|
||||||
throw new IOException("Disconnected");
|
|
||||||
}
|
|
||||||
// Process data
|
|
||||||
readBuffer.flip();
|
|
||||||
connection.processPackets(workerContext, packetProcessor);
|
connection.processPackets(workerContext, packetProcessor);
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
// TODO print exception? (should ignore disconnection)
|
// TODO print exception? (should ignore disconnection)
|
||||||
@ -124,11 +119,11 @@ public final class Worker {
|
|||||||
* Contains objects that we can be shared across all the connection of a {@link Worker worker}.
|
* Contains objects that we can be shared across all the connection of a {@link Worker worker}.
|
||||||
*/
|
*/
|
||||||
public static final class Context {
|
public static final class Context {
|
||||||
public final ByteBuffer readBuffer = ByteBuffer.allocateDirect(Server.SOCKET_BUFFER_SIZE);
|
public final BinaryBuffer readBuffer = BinaryBuffer.ofSize(Server.SOCKET_BUFFER_SIZE);
|
||||||
/**
|
/**
|
||||||
* Stores a single packet payload to be read.
|
* Stores a single packet payload to be read.
|
||||||
*/
|
*/
|
||||||
public final ByteBuffer contentBuffer = ByteBuffer.allocateDirect(Server.MAX_PACKET_SIZE);
|
public final BinaryBuffer contentBuffer = BinaryBuffer.ofSize(Server.MAX_PACKET_SIZE);
|
||||||
public final Inflater inflater = new Inflater();
|
public final Inflater inflater = new Inflater();
|
||||||
|
|
||||||
public void clearBuffers() {
|
public void clearBuffers() {
|
||||||
|
162
src/main/java/net/minestom/server/utils/binary/BinaryBuffer.java
Normal file
162
src/main/java/net/minestom/server/utils/binary/BinaryBuffer.java
Normal file
@ -0,0 +1,162 @@
|
|||||||
|
package net.minestom.server.utils.binary;
|
||||||
|
|
||||||
|
import org.jetbrains.annotations.ApiStatus;
|
||||||
|
import org.jetbrains.annotations.NotNull;
|
||||||
|
import org.jglrxavpok.hephaistos.nbt.NBTReader;
|
||||||
|
import org.jglrxavpok.hephaistos.nbt.NBTWriter;
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
import java.nio.channels.ReadableByteChannel;
|
||||||
|
import java.nio.channels.WritableByteChannel;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Manages off-heap memory.
|
||||||
|
* Not thread-safe.
|
||||||
|
*/
|
||||||
|
public final class BinaryBuffer {
|
||||||
|
private ByteBuffer nioBuffer; // To become a `MemorySegment` once released
|
||||||
|
private NBTReader nbtReader;
|
||||||
|
private NBTWriter nbtWriter;
|
||||||
|
|
||||||
|
private final int capacity;
|
||||||
|
private int readerOffset, writerOffset;
|
||||||
|
|
||||||
|
private BinaryBuffer(ByteBuffer buffer) {
|
||||||
|
this.nioBuffer = buffer;
|
||||||
|
this.capacity = buffer.capacity();
|
||||||
|
}
|
||||||
|
|
||||||
|
@ApiStatus.Internal
|
||||||
|
public static BinaryBuffer ofSize(int size) {
|
||||||
|
return new BinaryBuffer(ByteBuffer.allocateDirect(size));
|
||||||
|
}
|
||||||
|
|
||||||
|
public static BinaryBuffer copy(BinaryBuffer buffer) {
|
||||||
|
final int size = buffer.readableBytes();
|
||||||
|
final var temp = ByteBuffer.allocateDirect(size)
|
||||||
|
.put(buffer.asByteBuffer(0, size));
|
||||||
|
return new BinaryBuffer(temp);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void write(ByteBuffer buffer) {
|
||||||
|
final int size = buffer.remaining();
|
||||||
|
// TODO jdk 13 put with index
|
||||||
|
asByteBuffer(writerOffset, writerOffset + size).put(buffer);
|
||||||
|
this.writerOffset += size;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void write(BinaryBuffer buffer) {
|
||||||
|
write(buffer.asByteBuffer(buffer.readerOffset, buffer.writerOffset));
|
||||||
|
}
|
||||||
|
|
||||||
|
public int readVarInt() {
|
||||||
|
int value = 0;
|
||||||
|
final int maxRead = Math.min(5, readableBytes());
|
||||||
|
for (int i = 0; i < maxRead; i++) {
|
||||||
|
final int offset = readerOffset + i;
|
||||||
|
final byte k = nioBuffer.get(offset);
|
||||||
|
value |= (k & 0x7F) << i * 7;
|
||||||
|
if ((k & 0x80) != 128) {
|
||||||
|
this.readerOffset = offset + 1;
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.readerOffset += maxRead;
|
||||||
|
throw new RuntimeException("VarInt is too big");
|
||||||
|
}
|
||||||
|
|
||||||
|
public @NotNull Marker mark() {
|
||||||
|
return new Marker(readerOffset, writerOffset);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void reset(int readerOffset, int writerOffset) {
|
||||||
|
this.readerOffset = readerOffset;
|
||||||
|
this.writerOffset = writerOffset;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void reset(@NotNull Marker marker) {
|
||||||
|
reset(marker.readerOffset(), marker.writerOffset());
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean canWrite(int size) {
|
||||||
|
return writerOffset + size <= capacity;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int capacity() {
|
||||||
|
return capacity;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int readerOffset() {
|
||||||
|
return readerOffset;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int writerOffset() {
|
||||||
|
return writerOffset;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int readableBytes() {
|
||||||
|
return writerOffset - readerOffset;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void clear() {
|
||||||
|
this.readerOffset = 0;
|
||||||
|
this.writerOffset = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public ByteBuffer asByteBuffer(int reader, int writer) {
|
||||||
|
return nioBuffer.duplicate().position(reader).limit(writer);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void writeChannel(WritableByteChannel channel) throws IOException {
|
||||||
|
final int count = channel.write(asByteBuffer(readerOffset, writerOffset));
|
||||||
|
if (count == -1) {
|
||||||
|
// EOS
|
||||||
|
throw new IOException("Disconnected");
|
||||||
|
}
|
||||||
|
this.readerOffset += count;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void readChannel(ReadableByteChannel channel) throws IOException {
|
||||||
|
final int count = channel.read(asByteBuffer(readerOffset, capacity));
|
||||||
|
if (count == -1) {
|
||||||
|
// EOS
|
||||||
|
throw new IOException("Disconnected");
|
||||||
|
}
|
||||||
|
this.writerOffset += count;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "BinaryBuffer{" +
|
||||||
|
"readerOffset=" + readerOffset +
|
||||||
|
", writerOffset=" + writerOffset +
|
||||||
|
", capacity=" + capacity +
|
||||||
|
'}';
|
||||||
|
}
|
||||||
|
|
||||||
|
public static final class Marker {
|
||||||
|
private final int readerOffset, writerOffset;
|
||||||
|
|
||||||
|
private Marker(int readerOffset, int writerOffset) {
|
||||||
|
this.readerOffset = readerOffset;
|
||||||
|
this.writerOffset = writerOffset;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int readerOffset() {
|
||||||
|
return readerOffset;
|
||||||
|
}
|
||||||
|
|
||||||
|
public int writerOffset() {
|
||||||
|
return writerOffset;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "Marker{" +
|
||||||
|
"readerOffset=" + readerOffset +
|
||||||
|
", writerOffset=" + writerOffset +
|
||||||
|
'}';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user