HADOOP-11064. UnsatisifedLinkError with hadoop 2.4 JARs on hadoop-2.6 due to NativeCRC32 method changes. Contributed by Chris Nauroth.
(cherry picked from commit cbf0ae742a
)
This commit is contained in:
parent
30165d4cdd
commit
6e5194099c
|
@ -518,6 +518,9 @@ Release 2.6.0 - UNRELEASED
|
|||
wrong value if excluded nodes passed are not part of the cluster tree
|
||||
(vinayakumarb)
|
||||
|
||||
HADOOP-11064. UnsatisifedLinkError with hadoop 2.4 JARs on hadoop-2.6 due to
|
||||
NativeCRC32 method changes. (cnauroth)
|
||||
|
||||
Release 2.5.1 - 2014-09-05
|
||||
|
||||
INCOMPATIBLE CHANGES
|
||||
|
|
|
@ -21,6 +21,8 @@ import java.nio.ByteBuffer;
|
|||
|
||||
import org.apache.hadoop.fs.ChecksumException;
|
||||
|
||||
import com.google.common.annotations.VisibleForTesting;
|
||||
|
||||
/**
|
||||
* Wrapper around JNI support code to do checksum computation
|
||||
* natively.
|
||||
|
@ -87,6 +89,37 @@ class NativeCrc32 {
|
|||
"", 0, false);
|
||||
}
|
||||
|
||||
/**
|
||||
* Verify the given buffers of data and checksums, and throw an exception
|
||||
* if any checksum is invalid. The buffers given to this function should
|
||||
* have their position initially at the start of the data, and their limit
|
||||
* set at the end of the data. The position, limit, and mark are not
|
||||
* modified. This method is retained only for backwards-compatibility with
|
||||
* prior jar versions that need the corresponding JNI function.
|
||||
*
|
||||
* @param bytesPerSum the chunk size (eg 512 bytes)
|
||||
* @param checksumType the DataChecksum type constant
|
||||
* @param sums the DirectByteBuffer pointing at the beginning of the
|
||||
* stored checksums
|
||||
* @param sumsOffset start offset in sums buffer
|
||||
* @param data the DirectByteBuffer pointing at the beginning of the
|
||||
* data to check
|
||||
* @param dataOffset start offset in data buffer
|
||||
* @param dataLength length of data buffer
|
||||
* @param fileName the name of the file being verified
|
||||
* @param basePos the position in the file where the data buffer starts
|
||||
* @throws ChecksumException if there is an invalid checksum
|
||||
* @deprecated use {@link #nativeComputeChunkedSums(int, int, ByteBuffer, int,
|
||||
* ByteBuffer, int, int, String, long, boolean)} instead
|
||||
*/
|
||||
@Deprecated
|
||||
@VisibleForTesting
|
||||
static native void nativeVerifyChunkedSums(
|
||||
int bytesPerSum, int checksumType,
|
||||
ByteBuffer sums, int sumsOffset,
|
||||
ByteBuffer data, int dataOffset, int dataLength,
|
||||
String fileName, long basePos) throws ChecksumException;
|
||||
|
||||
private static native void nativeComputeChunkedSums(
|
||||
int bytesPerSum, int checksumType,
|
||||
ByteBuffer sums, int sumsOffset,
|
||||
|
|
|
@ -181,6 +181,18 @@ JNIEXPORT void JNICALL Java_org_apache_hadoop_util_NativeCrc32_nativeComputeChun
|
|||
}
|
||||
}
|
||||
|
||||
JNIEXPORT void JNICALL Java_org_apache_hadoop_util_NativeCrc32_nativeVerifyChunkedSums
|
||||
(JNIEnv *env, jclass clazz,
|
||||
jint bytes_per_checksum, jint j_crc_type,
|
||||
jobject j_sums, jint sums_offset,
|
||||
jobject j_data, jint data_offset, jint data_len,
|
||||
jstring j_filename, jlong base_pos)
|
||||
{
|
||||
Java_org_apache_hadoop_util_NativeCrc32_nativeComputeChunkedSums(env, clazz,
|
||||
bytes_per_checksum, j_crc_type, j_sums, sums_offset, j_data, data_offset,
|
||||
data_len, j_filename, base_pos, JNI_TRUE);
|
||||
}
|
||||
|
||||
JNIEXPORT void JNICALL Java_org_apache_hadoop_util_NativeCrc32_nativeComputeChunkedSumsByteArray
|
||||
(JNIEnv *env, jclass clazz,
|
||||
jint bytes_per_checksum, jint j_crc_type,
|
||||
|
|
|
@ -0,0 +1,229 @@
|
|||
/**
|
||||
* Licensed to the Apache Software Foundation (ASF) under one
|
||||
* or more contributor license agreements. See the NOTICE file
|
||||
* distributed with this work for additional information
|
||||
* regarding copyright ownership. The ASF 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 org.apache.hadoop.util;
|
||||
|
||||
import static org.junit.Assert.*;
|
||||
import static org.junit.Assume.*;
|
||||
|
||||
import java.nio.ByteBuffer;
|
||||
import java.util.ArrayList;
|
||||
import java.util.Collection;
|
||||
|
||||
import org.apache.hadoop.conf.Configuration;
|
||||
import org.apache.hadoop.fs.ChecksumException;
|
||||
|
||||
import org.junit.Before;
|
||||
import org.junit.Rule;
|
||||
import org.junit.Test;
|
||||
import org.junit.rules.ExpectedException;
|
||||
import org.junit.runner.RunWith;
|
||||
import org.junit.runners.Parameterized;
|
||||
import org.junit.runners.Parameterized.Parameters;
|
||||
|
||||
@RunWith(Parameterized.class)
|
||||
public class TestNativeCrc32 {
|
||||
|
||||
private static final long BASE_POSITION = 0;
|
||||
private static final int IO_BYTES_PER_CHECKSUM_DEFAULT = 512;
|
||||
private static final String IO_BYTES_PER_CHECKSUM_KEY =
|
||||
"io.bytes.per.checksum";
|
||||
private static final int NUM_CHUNKS = 3;
|
||||
|
||||
private final DataChecksum.Type checksumType;
|
||||
|
||||
private int bytesPerChecksum;
|
||||
private String fileName;
|
||||
private ByteBuffer data, checksums;
|
||||
private DataChecksum checksum;
|
||||
|
||||
@Rule
|
||||
public ExpectedException exception = ExpectedException.none();
|
||||
|
||||
@Parameters
|
||||
public static Collection<Object[]> data() {
|
||||
Collection<Object[]> params = new ArrayList<Object[]>(2);
|
||||
params.add(new Object[] { DataChecksum.Type.CRC32 });
|
||||
params.add(new Object[] { DataChecksum.Type.CRC32C });
|
||||
return params;
|
||||
}
|
||||
|
||||
public TestNativeCrc32(DataChecksum.Type checksumType) {
|
||||
this.checksumType = checksumType;
|
||||
}
|
||||
|
||||
@Before
|
||||
public void setup() {
|
||||
assumeTrue(NativeCrc32.isAvailable());
|
||||
assertEquals(
|
||||
"These tests assume they can write a checksum value as a 4-byte int.", 4,
|
||||
checksumType.size);
|
||||
Configuration conf = new Configuration();
|
||||
bytesPerChecksum = conf.getInt(IO_BYTES_PER_CHECKSUM_KEY,
|
||||
IO_BYTES_PER_CHECKSUM_DEFAULT);
|
||||
fileName = this.getClass().getSimpleName();
|
||||
checksum = DataChecksum.newDataChecksum(checksumType, bytesPerChecksum);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testVerifyChunkedSumsSuccess() throws ChecksumException {
|
||||
allocateDirectByteBuffers();
|
||||
fillDataAndValidChecksums();
|
||||
NativeCrc32.verifyChunkedSums(bytesPerChecksum, checksumType.id,
|
||||
checksums, data, fileName, BASE_POSITION);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testVerifyChunkedSumsFail() throws ChecksumException {
|
||||
allocateDirectByteBuffers();
|
||||
fillDataAndInvalidChecksums();
|
||||
exception.expect(ChecksumException.class);
|
||||
NativeCrc32.verifyChunkedSums(bytesPerChecksum, checksumType.id,
|
||||
checksums, data, fileName, BASE_POSITION);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testVerifyChunkedSumsByteArraySuccess() throws ChecksumException {
|
||||
allocateArrayByteBuffers();
|
||||
fillDataAndValidChecksums();
|
||||
NativeCrc32.verifyChunkedSumsByteArray(bytesPerChecksum, checksumType.id,
|
||||
checksums.array(), checksums.position(), data.array(), data.position(),
|
||||
data.remaining(), fileName, BASE_POSITION);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testVerifyChunkedSumsByteArrayFail() throws ChecksumException {
|
||||
allocateArrayByteBuffers();
|
||||
fillDataAndInvalidChecksums();
|
||||
exception.expect(ChecksumException.class);
|
||||
NativeCrc32.verifyChunkedSumsByteArray(bytesPerChecksum, checksumType.id,
|
||||
checksums.array(), checksums.position(), data.array(), data.position(),
|
||||
data.remaining(), fileName, BASE_POSITION);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testCalculateChunkedSumsSuccess() throws ChecksumException {
|
||||
allocateDirectByteBuffers();
|
||||
fillDataAndValidChecksums();
|
||||
NativeCrc32.calculateChunkedSums(bytesPerChecksum, checksumType.id,
|
||||
checksums, data);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testCalculateChunkedSumsFail() throws ChecksumException {
|
||||
allocateDirectByteBuffers();
|
||||
fillDataAndInvalidChecksums();
|
||||
NativeCrc32.calculateChunkedSums(bytesPerChecksum, checksumType.id,
|
||||
checksums, data);
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testCalculateChunkedSumsByteArraySuccess() throws ChecksumException {
|
||||
allocateArrayByteBuffers();
|
||||
fillDataAndValidChecksums();
|
||||
NativeCrc32.calculateChunkedSumsByteArray(bytesPerChecksum, checksumType.id,
|
||||
checksums.array(), checksums.position(), data.array(), data.position(),
|
||||
data.remaining());
|
||||
}
|
||||
|
||||
@Test
|
||||
public void testCalculateChunkedSumsByteArrayFail() throws ChecksumException {
|
||||
allocateArrayByteBuffers();
|
||||
fillDataAndInvalidChecksums();
|
||||
NativeCrc32.calculateChunkedSumsByteArray(bytesPerChecksum, checksumType.id,
|
||||
checksums.array(), checksums.position(), data.array(), data.position(),
|
||||
data.remaining());
|
||||
}
|
||||
|
||||
@Test
|
||||
@SuppressWarnings("deprecation")
|
||||
public void testNativeVerifyChunkedSumsSuccess() throws ChecksumException {
|
||||
allocateDirectByteBuffers();
|
||||
fillDataAndValidChecksums();
|
||||
NativeCrc32.nativeVerifyChunkedSums(bytesPerChecksum, checksumType.id,
|
||||
checksums, checksums.position(), data, data.position(), data.remaining(),
|
||||
fileName, BASE_POSITION);
|
||||
}
|
||||
|
||||
@Test
|
||||
@SuppressWarnings("deprecation")
|
||||
public void testNativeVerifyChunkedSumsFail() throws ChecksumException {
|
||||
allocateDirectByteBuffers();
|
||||
fillDataAndInvalidChecksums();
|
||||
exception.expect(ChecksumException.class);
|
||||
NativeCrc32.nativeVerifyChunkedSums(bytesPerChecksum, checksumType.id,
|
||||
checksums, checksums.position(), data, data.position(), data.remaining(),
|
||||
fileName, BASE_POSITION);
|
||||
}
|
||||
|
||||
/**
|
||||
* Allocates data buffer and checksums buffer as arrays on the heap.
|
||||
*/
|
||||
private void allocateArrayByteBuffers() {
|
||||
data = ByteBuffer.wrap(new byte[bytesPerChecksum * NUM_CHUNKS]);
|
||||
checksums = ByteBuffer.wrap(new byte[NUM_CHUNKS * checksumType.size]);
|
||||
}
|
||||
|
||||
/**
|
||||
* Allocates data buffer and checksums buffer as direct byte buffers.
|
||||
*/
|
||||
private void allocateDirectByteBuffers() {
|
||||
data = ByteBuffer.allocateDirect(bytesPerChecksum * NUM_CHUNKS);
|
||||
checksums = ByteBuffer.allocateDirect(NUM_CHUNKS * checksumType.size);
|
||||
}
|
||||
|
||||
/**
|
||||
* Fill data buffer with monotonically increasing byte values. Overflow is
|
||||
* fine, because it's just test data. Update the checksum with the same byte
|
||||
* values. After every chunk, write the checksum to the checksums buffer.
|
||||
* After finished writing, flip the buffers to prepare them for reading.
|
||||
*/
|
||||
private void fillDataAndValidChecksums() {
|
||||
for (int i = 0; i < NUM_CHUNKS; ++i) {
|
||||
for (int j = 0; j < bytesPerChecksum; ++j) {
|
||||
byte b = (byte)((i * bytesPerChecksum + j) & 0xFF);
|
||||
data.put(b);
|
||||
checksum.update(b);
|
||||
}
|
||||
checksums.putInt((int)checksum.getValue());
|
||||
checksum.reset();
|
||||
}
|
||||
data.flip();
|
||||
checksums.flip();
|
||||
}
|
||||
|
||||
/**
|
||||
* Fill data buffer with monotonically increasing byte values. Overflow is
|
||||
* fine, because it's just test data. Update the checksum with different byte
|
||||
* byte values, so that the checksums are incorrect intentionally. After every
|
||||
* chunk, write the checksum to the checksums buffer. After finished writing,
|
||||
* flip the buffers to prepare them for reading.
|
||||
*/
|
||||
private void fillDataAndInvalidChecksums() {
|
||||
for (int i = 0; i < NUM_CHUNKS; ++i) {
|
||||
for (int j = 0; j < bytesPerChecksum; ++j) {
|
||||
byte b = (byte)((i * bytesPerChecksum + j) & 0xFF);
|
||||
data.put(b);
|
||||
checksum.update((byte)(b + 1));
|
||||
}
|
||||
checksums.putInt((int)checksum.getValue());
|
||||
checksum.reset();
|
||||
}
|
||||
data.flip();
|
||||
checksums.flip();
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue