Only try to calculate direct memory offset when sun.misc.Unsafe is present

Motivation:

We should only try to calculate the direct memory offset when sun.misc.Unsafe is present as otherwise it will fail with an NPE as PlatformDependent.directBufferAddress(...) will throw it.
This problem was introduced by 66b9be3a46.

Modifications:

Use offset of 0 if no sun.misc.Unsafe is present.

Result:

PooledByteBufAllocator also works again when no sun.misc.Unsafe is present.
This commit is contained in:
Norman Maurer 2017-02-13 07:42:22 +01:00
parent d8596d2d90
commit 54339c08ac
3 changed files with 16 additions and 1 deletions

View File

@ -722,7 +722,10 @@ abstract class PoolArena<T> implements PoolArenaMetric {
} }
private int offsetCacheLine(ByteBuffer memory) { private int offsetCacheLine(ByteBuffer memory) {
return (int) (PlatformDependent.directBufferAddress(memory) & directMemoryCacheAlignmentMask); // We can only calculate the offset if Unsafe is present as otherwise directBufferAddress(...) will
// throw an NPE.
return HAS_UNSAFE ?
(int) (PlatformDependent.directBufferAddress(memory) & directMemoryCacheAlignmentMask) : 0;
} }
@Override @Override

View File

@ -212,6 +212,9 @@ public class PooledByteBufAllocator extends AbstractByteBufAllocator {
throw new IllegalArgumentException("directMemoryCacheAlignment: " throw new IllegalArgumentException("directMemoryCacheAlignment: "
+ directMemoryCacheAlignment + " (expected: >= 0)"); + directMemoryCacheAlignment + " (expected: >= 0)");
} }
if (directMemoryCacheAlignment > 0 && !isDirectMemoryCacheAlignmentSupported()) {
throw new IllegalArgumentException("directMemoryCacheAlignment is not supported");
}
if ((directMemoryCacheAlignment & -directMemoryCacheAlignment) != directMemoryCacheAlignment) { if ((directMemoryCacheAlignment & -directMemoryCacheAlignment) != directMemoryCacheAlignment) {
throw new IllegalArgumentException("directMemoryCacheAlignment: " throw new IllegalArgumentException("directMemoryCacheAlignment: "
@ -370,6 +373,13 @@ public class PooledByteBufAllocator extends AbstractByteBufAllocator {
return DEFAULT_NORMAL_CACHE_SIZE; return DEFAULT_NORMAL_CACHE_SIZE;
} }
/**
* Return {@code true} if direct memory cache aligment is supported, {@code false} otherwise.
*/
public static boolean isDirectMemoryCacheAlignmentSupported() {
return PlatformDependent.hasUnsafe();
}
@Override @Override
public boolean isDirectBufferPooled() { public boolean isDirectBufferPooled() {
return directArenas != null; return directArenas != null;

View File

@ -19,6 +19,7 @@ package io.netty.buffer;
import io.netty.util.concurrent.FastThreadLocal; import io.netty.util.concurrent.FastThreadLocal;
import io.netty.util.concurrent.FastThreadLocalThread; import io.netty.util.concurrent.FastThreadLocalThread;
import io.netty.util.internal.SystemPropertyUtil; import io.netty.util.internal.SystemPropertyUtil;
import org.junit.Assume;
import org.junit.Test; import org.junit.Test;
import java.util.ArrayList; import java.util.ArrayList;
@ -59,6 +60,7 @@ public class PooledByteBufAllocatorTest extends AbstractByteBufAllocatorTest {
@Test @Test
public void testArenaMetricsCacheAlign() { public void testArenaMetricsCacheAlign() {
Assume.assumeTrue(PooledByteBufAllocator.isDirectMemoryCacheAlignmentSupported());
testArenaMetrics0(new PooledByteBufAllocator(true, 2, 2, 8192, 11, 1000, 1000, 1000, true, 64), 100, 1, 1, 0); testArenaMetrics0(new PooledByteBufAllocator(true, 2, 2, 8192, 11, 1000, 1000, 1000, true, 64), 100, 1, 1, 0);
} }