From 57cb7a8a91a25f94068605eca136949f6b3d1226 Mon Sep 17 00:00:00 2001 From: Chris Vest Date: Thu, 29 Oct 2020 10:35:47 +0100 Subject: [PATCH] Fix explicitly little-endian accessors in SwappedByteBuf (#10747) Motivation: Some buffers implement ByteBuf#order(order) by wrapping themselves in a SwappedByteBuf. The SwappedByteBuf is then responsible for swapping the byte order on accesses. The explicitly little-endian accessor methods, however, should not be swapped to big-endian, but instead remain explicitly little-endian. Modification: The SwappedByteBuf was passing through calls to e.g. writeIntLE, to the big-endian equivalent, e.g. writeInt. This has been changed so that these calls delegate to their explicitly little-endian counterpart. Result: This makes all buffers that make use of SwappedByteBuf for their endian-ness configuration, consistent with all the buffers that use other implementation strategies. In the end, all buffers now behave exactly the same, when using their explicitly little-endian accessor methods. --- .../java/io/netty/buffer/ByteBufUtil.java | 9 ++- .../java/io/netty/buffer/SwappedByteBuf.java | 16 ++--- .../io/netty/buffer/AbstractByteBufTest.java | 59 +++++++++++++++++++ .../java/io/netty/buffer/ByteBufUtilTest.java | 6 +- 4 files changed, 76 insertions(+), 14 deletions(-) diff --git a/buffer/src/main/java/io/netty/buffer/ByteBufUtil.java b/buffer/src/main/java/io/netty/buffer/ByteBufUtil.java index 584ad50471..532cd6b3bb 100644 --- a/buffer/src/main/java/io/netty/buffer/ByteBufUtil.java +++ b/buffer/src/main/java/io/netty/buffer/ByteBufUtil.java @@ -437,7 +437,8 @@ public final class ByteBufUtil { */ @SuppressWarnings("deprecation") public static ByteBuf writeShortBE(ByteBuf buf, int shortValue) { - return buf.order() == ByteOrder.BIG_ENDIAN? buf.writeShort(shortValue) : buf.writeShortLE(shortValue); + return buf.order() == ByteOrder.BIG_ENDIAN? buf.writeShort(shortValue) : + buf.writeShort(swapShort((short) shortValue)); } /** @@ -445,7 +446,8 @@ public final class ByteBufUtil { */ @SuppressWarnings("deprecation") public static ByteBuf setShortBE(ByteBuf buf, int index, int shortValue) { - return buf.order() == ByteOrder.BIG_ENDIAN? buf.setShort(index, shortValue) : buf.setShortLE(index, shortValue); + return buf.order() == ByteOrder.BIG_ENDIAN? buf.setShort(index, shortValue) : + buf.setShort(index, swapShort((short) shortValue)); } /** @@ -453,7 +455,8 @@ public final class ByteBufUtil { */ @SuppressWarnings("deprecation") public static ByteBuf writeMediumBE(ByteBuf buf, int mediumValue) { - return buf.order() == ByteOrder.BIG_ENDIAN? buf.writeMedium(mediumValue) : buf.writeMediumLE(mediumValue); + return buf.order() == ByteOrder.BIG_ENDIAN? buf.writeMedium(mediumValue) : + buf.writeMedium(swapMedium(mediumValue)); } /** diff --git a/buffer/src/main/java/io/netty/buffer/SwappedByteBuf.java b/buffer/src/main/java/io/netty/buffer/SwappedByteBuf.java index 06773d7736..0650c99ba8 100644 --- a/buffer/src/main/java/io/netty/buffer/SwappedByteBuf.java +++ b/buffer/src/main/java/io/netty/buffer/SwappedByteBuf.java @@ -522,7 +522,7 @@ public class SwappedByteBuf extends ByteBuf { @Override public short readShortLE() { - return buf.readShort(); + return buf.readShortLE(); } @Override @@ -542,7 +542,7 @@ public class SwappedByteBuf extends ByteBuf { @Override public int readMediumLE() { - return buf.readMedium(); + return buf.readMediumLE(); } @Override @@ -562,7 +562,7 @@ public class SwappedByteBuf extends ByteBuf { @Override public int readIntLE() { - return buf.readInt(); + return buf.readIntLE(); } @Override @@ -582,7 +582,7 @@ public class SwappedByteBuf extends ByteBuf { @Override public long readLongLE() { - return buf.readLong(); + return buf.readLongLE(); } @Override @@ -698,7 +698,7 @@ public class SwappedByteBuf extends ByteBuf { @Override public ByteBuf writeShortLE(int value) { - buf.writeShort((short) value); + buf.writeShortLE((short) value); return this; } @@ -710,7 +710,7 @@ public class SwappedByteBuf extends ByteBuf { @Override public ByteBuf writeMediumLE(int value) { - buf.writeMedium(value); + buf.writeMediumLE(value); return this; } @@ -722,7 +722,7 @@ public class SwappedByteBuf extends ByteBuf { @Override public ByteBuf writeIntLE(int value) { - buf.writeInt(value); + buf.writeIntLE(value); return this; } @@ -734,7 +734,7 @@ public class SwappedByteBuf extends ByteBuf { @Override public ByteBuf writeLongLE(long value) { - buf.writeLong(value); + buf.writeLongLE(value); return this; } diff --git a/buffer/src/test/java/io/netty/buffer/AbstractByteBufTest.java b/buffer/src/test/java/io/netty/buffer/AbstractByteBufTest.java index 9d314b2cfc..c7e0f18446 100644 --- a/buffer/src/test/java/io/netty/buffer/AbstractByteBufTest.java +++ b/buffer/src/test/java/io/netty/buffer/AbstractByteBufTest.java @@ -4907,4 +4907,63 @@ public abstract class AbstractByteBufTest { buffer.release(); } } + + @Test + public void testEndiannessIndexOf() { + buffer.clear(); + final int v = 0x02030201; + buffer.writeIntLE(v); + buffer.writeByte(0x01); + + assertEquals(-1, buffer.indexOf(1, 4, (byte) 1)); + assertEquals(-1, buffer.indexOf(4, 1, (byte) 1)); + assertEquals(1, buffer.indexOf(1, 4, (byte) 2)); + assertEquals(3, buffer.indexOf(4, 1, (byte) 2)); + } + + @Test + public void explicitLittleEndianReadMethodsMustAlwaysUseLittleEndianByteOrder() { + buffer.clear(); + buffer.writeBytes(new byte[] {0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07, 0x08}); + assertEquals(0x0201, buffer.readShortLE()); + buffer.readerIndex(0); + assertEquals(0x0201, buffer.readUnsignedShortLE()); + buffer.readerIndex(0); + assertEquals(0x030201, buffer.readMediumLE()); + buffer.readerIndex(0); + assertEquals(0x030201, buffer.readUnsignedMediumLE()); + buffer.readerIndex(0); + assertEquals(0x04030201, buffer.readIntLE()); + buffer.readerIndex(0); + assertEquals(0x04030201, buffer.readUnsignedIntLE()); + buffer.readerIndex(0); + assertEquals(0x04030201, Float.floatToRawIntBits(buffer.readFloatLE())); + buffer.readerIndex(0); + assertEquals(0x0807060504030201L, buffer.readLongLE()); + buffer.readerIndex(0); + assertEquals(0x0807060504030201L, Double.doubleToRawLongBits(buffer.readDoubleLE())); + buffer.readerIndex(0); + } + + @Test + public void explicitLittleEndianWriteMethodsMustAlwaysUseLittleEndianByteOrder() { + buffer.clear(); + buffer.writeShortLE(0x0102); + assertEquals(0x0102, buffer.readShortLE()); + buffer.clear(); + buffer.writeMediumLE(0x010203); + assertEquals(0x010203, buffer.readMediumLE()); + buffer.clear(); + buffer.writeIntLE(0x01020304); + assertEquals(0x01020304, buffer.readIntLE()); + buffer.clear(); + buffer.writeFloatLE(Float.intBitsToFloat(0x01020304)); + assertEquals(0x01020304, Float.floatToRawIntBits(buffer.readFloatLE())); + buffer.clear(); + buffer.writeLongLE(0x0102030405060708L); + assertEquals(0x0102030405060708L, buffer.readLongLE()); + buffer.clear(); + buffer.writeDoubleLE(Double.longBitsToDouble(0x0102030405060708L)); + assertEquals(0x0102030405060708L, Double.doubleToRawLongBits(buffer.readDoubleLE())); + } } diff --git a/buffer/src/test/java/io/netty/buffer/ByteBufUtilTest.java b/buffer/src/test/java/io/netty/buffer/ByteBufUtilTest.java index 026c312890..43267191f2 100644 --- a/buffer/src/test/java/io/netty/buffer/ByteBufUtilTest.java +++ b/buffer/src/test/java/io/netty/buffer/ByteBufUtilTest.java @@ -189,7 +189,7 @@ public class ByteBufUtilTest { buf = buffer(2).order(ByteOrder.LITTLE_ENDIAN); ByteBufUtil.writeShortBE(buf, expected); - assertEquals((short) expected, buf.readShortLE()); + assertEquals(ByteBufUtil.swapShort((short) expected), buf.readShortLE()); buf.readerIndex(0); assertEquals(ByteBufUtil.swapShort((short) expected), buf.readShort()); buf.release(); @@ -209,7 +209,7 @@ public class ByteBufUtilTest { buf = Unpooled.wrappedBuffer(new byte[2]).order(ByteOrder.LITTLE_ENDIAN); ByteBufUtil.setShortBE(buf, 0, shortValue); - assertEquals((short) shortValue, buf.readShortLE()); + assertEquals(ByteBufUtil.swapShort((short) shortValue), buf.readShortLE()); buf.readerIndex(0); assertEquals(ByteBufUtil.swapShort((short) shortValue), buf.readShort()); buf.release(); @@ -229,7 +229,7 @@ public class ByteBufUtilTest { buf = buffer(4).order(ByteOrder.LITTLE_ENDIAN); ByteBufUtil.writeMediumBE(buf, mediumValue); - assertEquals(mediumValue, buf.readMediumLE()); + assertEquals(ByteBufUtil.swapMedium(mediumValue), buf.readMediumLE()); buf.readerIndex(0); assertEquals(ByteBufUtil.swapMedium(mediumValue), buf.readMedium()); buf.release();