Fix ReflectiveByteBufChecksum with direct buffers (#9244)

Motivation:

ReflectiveByteBufChecksum#update(buf, off, len) ignores provided offset
and length arguments when operating on direct buffers, leading to wrong
byte sequences being checksummed and ultimately incorrect checksum
values (unless checksumming the entire buffer).

Modifications:

Use the provided offset and length arguments to get the correct nio
buffer to checksum; add test coverage exercising the four meaningfully
different offset and length combinations.

Result:

Offset and length are respected and a correct checksum gets calculated;
simple unit test should prevent regressions in the future.
This commit is contained in:
Aleksey Yeschenko 2019-06-17 15:32:13 +01:00 committed by Norman Maurer
parent 96feca1d23
commit a2583d0d3c
3 changed files with 88 additions and 1 deletions

View File

@ -100,7 +100,7 @@ abstract class ByteBufChecksum implements Checksum {
update(b.array(), b.arrayOffset() + off, len);
} else {
try {
method.invoke(checksum, CompressionUtil.safeNioBuffer(b));
method.invoke(checksum, CompressionUtil.safeNioBuffer(b, off, len));
} catch (Throwable cause) {
throw new Error();
}

View File

@ -40,4 +40,9 @@ final class CompressionUtil {
return buffer.nioBufferCount() == 1 ? buffer.internalNioBuffer(buffer.readerIndex(), buffer.readableBytes())
: buffer.nioBuffer();
}
static ByteBuffer safeNioBuffer(ByteBuf buffer, int index, int length) {
return buffer.nioBufferCount() == 1 ? buffer.internalNioBuffer(index, length)
: buffer.nioBuffer(index, length);
}
}

View File

@ -0,0 +1,82 @@
/*
* Copyright 2019 The Netty Project
*
* The Netty Project licenses this file to you under the Apache License,
* version 2.0 (the "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at:
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*/
package io.netty.handler.codec.compression;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.Unpooled;
import net.jpountz.xxhash.XXHashFactory;
import org.junit.BeforeClass;
import org.junit.Test;
import java.util.Random;
import java.util.zip.Adler32;
import java.util.zip.CRC32;
import java.util.zip.Checksum;
import static org.junit.Assert.*;
public class ByteBufChecksumTest {
private static final byte[] BYTE_ARRAY = new byte[1024];
@BeforeClass
public static void setUp() {
new Random().nextBytes(BYTE_ARRAY);
}
@Test
public void testHeapByteBufUpdate() {
testUpdate(Unpooled.wrappedBuffer(BYTE_ARRAY));
}
@Test
public void testDirectByteBufUpdate() {
ByteBuf buf = Unpooled.directBuffer(BYTE_ARRAY.length);
buf.writeBytes(BYTE_ARRAY);
testUpdate(buf);
}
private static void testUpdate(ByteBuf buf) {
try {
testUpdate(xxHash32(), ByteBufChecksum.wrapChecksum(xxHash32()), buf);
testUpdate(new CRC32(), ByteBufChecksum.wrapChecksum(new CRC32()), buf);
testUpdate(new Adler32(), ByteBufChecksum.wrapChecksum(new Adler32()), buf);
} finally {
buf.release();
}
}
private static void testUpdate(Checksum checksum, ByteBufChecksum wrapped, ByteBuf buf) {
testUpdate(checksum, wrapped, buf, 0, BYTE_ARRAY.length);
testUpdate(checksum, wrapped, buf, 0, BYTE_ARRAY.length - 1);
testUpdate(checksum, wrapped, buf, 1, BYTE_ARRAY.length - 1);
testUpdate(checksum, wrapped, buf, 1, BYTE_ARRAY.length - 2);
}
private static void testUpdate(Checksum checksum, ByteBufChecksum wrapped, ByteBuf buf, int off, int len) {
checksum.reset();
wrapped.reset();
checksum.update(BYTE_ARRAY, off, len);
wrapped.update(buf, off, len);
assertEquals(checksum.getValue(), wrapped.getValue());
}
private static Checksum xxHash32() {
return XXHashFactory.fastestInstance().newStreamingHash32(Lz4Constants.DEFAULT_SEED).asChecksum();
}
}