Extract ChannelFinder into separate top level classes
This commit is contained in:
parent
e55a1f11b5
commit
fe4a8d6dc9
@ -0,0 +1,5 @@
|
|||||||
|
package io.netty.channel.socket.aio;
|
||||||
|
|
||||||
|
interface AioChannelFinder {
|
||||||
|
AbstractAioChannel findChannel(Runnable command) throws Exception;
|
||||||
|
}
|
@ -15,47 +15,39 @@
|
|||||||
*/
|
*/
|
||||||
package io.netty.channel.socket.aio;
|
package io.netty.channel.socket.aio;
|
||||||
|
|
||||||
|
import io.netty.channel.ChannelTaskScheduler;
|
||||||
import io.netty.channel.EventExecutor;
|
import io.netty.channel.EventExecutor;
|
||||||
import io.netty.channel.EventLoopException;
|
import io.netty.channel.EventLoopException;
|
||||||
import io.netty.channel.MultithreadEventLoopGroup;
|
import io.netty.channel.MultithreadEventLoopGroup;
|
||||||
import io.netty.channel.ChannelTaskScheduler;
|
|
||||||
import io.netty.logging.InternalLogger;
|
import io.netty.logging.InternalLogger;
|
||||||
import io.netty.logging.InternalLoggerFactory;
|
import io.netty.logging.InternalLoggerFactory;
|
||||||
import io.netty.util.internal.DetectionUtil;
|
import io.netty.util.internal.DetectionUtil;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.lang.reflect.Field;
|
|
||||||
import java.nio.channels.AsynchronousChannelGroup;
|
import java.nio.channels.AsynchronousChannelGroup;
|
||||||
import java.util.ArrayDeque;
|
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.Deque;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.concurrent.AbstractExecutorService;
|
import java.util.concurrent.AbstractExecutorService;
|
||||||
import java.util.concurrent.ConcurrentHashMap;
|
|
||||||
import java.util.concurrent.ConcurrentMap;
|
|
||||||
import java.util.concurrent.ThreadFactory;
|
import java.util.concurrent.ThreadFactory;
|
||||||
import java.util.concurrent.TimeUnit;
|
import java.util.concurrent.TimeUnit;
|
||||||
|
|
||||||
import sun.misc.Unsafe;
|
|
||||||
|
|
||||||
public class AioEventLoopGroup extends MultithreadEventLoopGroup {
|
public class AioEventLoopGroup extends MultithreadEventLoopGroup {
|
||||||
private static final InternalLogger LOGGER = InternalLoggerFactory.getInstance(AioEventLoopGroup.class);
|
private static final InternalLogger LOGGER = InternalLoggerFactory.getInstance(AioEventLoopGroup.class);
|
||||||
private static final ChannelFinder CHANNEL_FINDER;
|
private static final AioChannelFinder CHANNEL_FINDER;
|
||||||
|
|
||||||
static {
|
static {
|
||||||
ChannelFinder finder;
|
AioChannelFinder finder;
|
||||||
try {
|
try {
|
||||||
// check if Unsafe is present on the classpath
|
|
||||||
// and if so try to instance the UnsafeChannelFinder
|
|
||||||
// too get the optimal speed.
|
|
||||||
if (DetectionUtil.hasUnsafe()) {
|
if (DetectionUtil.hasUnsafe()) {
|
||||||
finder = new UnsafeChannelFinder();
|
finder = new UnsafeAioChannelFinder();
|
||||||
} else {
|
} else {
|
||||||
finder = new ReflectionChannelFinder();
|
finder = new DefaultAioChannelFinder();
|
||||||
}
|
}
|
||||||
} catch (Throwable t) {
|
} catch (Throwable t) {
|
||||||
LOGGER.debug("Unable to instance UnsafeChannelFinder fallback to ReflectionChannelFinder", t);
|
LOGGER.debug(String.format(
|
||||||
finder = new ReflectionChannelFinder();
|
"Unable to instance the optimal %s implementation - falling back to %s.",
|
||||||
|
AioChannelFinder.class.getSimpleName(), DefaultAioChannelFinder.class.getSimpleName()), t);
|
||||||
|
finder = new DefaultAioChannelFinder();
|
||||||
}
|
}
|
||||||
CHANNEL_FINDER = finder;
|
CHANNEL_FINDER = finder;
|
||||||
}
|
}
|
||||||
@ -145,102 +137,4 @@ public class AioEventLoopGroup extends MultithreadEventLoopGroup {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
interface ChannelFinder {
|
|
||||||
AbstractAioChannel findChannel(Runnable command) throws Exception;
|
|
||||||
}
|
|
||||||
|
|
||||||
static class ReflectionChannelFinder implements ChannelFinder {
|
|
||||||
private static final ConcurrentMap<Class<?>, Field[]> fieldCache = new ConcurrentHashMap<Class<?>, Field[]>();
|
|
||||||
private static final Field[] FAILURE = new Field[0];
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public AbstractAioChannel findChannel(Runnable command) throws Exception {
|
|
||||||
Class<?> commandType = command.getClass();
|
|
||||||
Field[] fields = fieldCache.get(commandType);
|
|
||||||
if (fields == null) {
|
|
||||||
try {
|
|
||||||
fields = findFieldSequence(command, new ArrayDeque<Field>(2));
|
|
||||||
} catch (Throwable t) {
|
|
||||||
// Failed to get the field list
|
|
||||||
}
|
|
||||||
|
|
||||||
if (fields == null) {
|
|
||||||
fields = FAILURE;
|
|
||||||
}
|
|
||||||
|
|
||||||
fieldCache.put(commandType, fields); // No need to use putIfAbsent()
|
|
||||||
}
|
|
||||||
|
|
||||||
if (fields == FAILURE) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
final int lastIndex = fields.length - 1;
|
|
||||||
for (int i = 0; i < lastIndex; i ++) {
|
|
||||||
command = (Runnable) get(fields[i], command);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (AbstractAioChannel) get(fields[lastIndex], command);
|
|
||||||
}
|
|
||||||
|
|
||||||
private Field[] findFieldSequence(Runnable command, Deque<Field> fields) throws Exception {
|
|
||||||
Class<?> commandType = command.getClass();
|
|
||||||
for (Field f: commandType.getDeclaredFields()) {
|
|
||||||
if (f.getType() == Runnable.class) {
|
|
||||||
f.setAccessible(true);
|
|
||||||
fields.addLast(f);
|
|
||||||
try {
|
|
||||||
Field[] ret = findFieldSequence((Runnable) get(f, command), fields);
|
|
||||||
if (ret != null) {
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
fields.removeLast();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (f.getType() == Object.class) {
|
|
||||||
f.setAccessible(true);
|
|
||||||
fields.addLast(f);
|
|
||||||
try {
|
|
||||||
Object candidate = get(f, command);
|
|
||||||
if (candidate instanceof AbstractAioChannel) {
|
|
||||||
return fields.toArray(new Field[fields.size()]);
|
|
||||||
}
|
|
||||||
} finally {
|
|
||||||
fields.removeLast();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected Object get(Field f, Object command) throws Exception {
|
|
||||||
return f.get(command);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
static class UnsafeChannelFinder extends ReflectionChannelFinder {
|
|
||||||
private static final Unsafe UNSAFE = getUnsafe();
|
|
||||||
|
|
||||||
@Override
|
|
||||||
protected Object get(Field f, Object command) throws Exception {
|
|
||||||
// using Unsafe to directly access the field. This should be
|
|
||||||
// faster then "pure" reflection
|
|
||||||
long offset = UNSAFE.objectFieldOffset(f);
|
|
||||||
return UNSAFE.getObject(command, offset);
|
|
||||||
}
|
|
||||||
|
|
||||||
private static Unsafe getUnsafe() {
|
|
||||||
try {
|
|
||||||
Field singleoneInstanceField = Unsafe.class.getDeclaredField("theUnsafe");
|
|
||||||
singleoneInstanceField.setAccessible(true);
|
|
||||||
return (Unsafe) singleoneInstanceField.get(null);
|
|
||||||
} catch (Throwable cause) {
|
|
||||||
throw new RuntimeException("Error while obtaining sun.misc.Unsafe", cause);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,79 @@
|
|||||||
|
package io.netty.channel.socket.aio;
|
||||||
|
|
||||||
|
import java.lang.reflect.Field;
|
||||||
|
import java.util.ArrayDeque;
|
||||||
|
import java.util.Deque;
|
||||||
|
import java.util.concurrent.ConcurrentHashMap;
|
||||||
|
import java.util.concurrent.ConcurrentMap;
|
||||||
|
|
||||||
|
class DefaultAioChannelFinder implements AioChannelFinder {
|
||||||
|
private static final ConcurrentMap<Class<?>, Field[]> fieldCache = new ConcurrentHashMap<Class<?>, Field[]>();
|
||||||
|
private static final Field[] FAILURE = new Field[0];
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public AbstractAioChannel findChannel(Runnable command) throws Exception {
|
||||||
|
Class<?> commandType = command.getClass();
|
||||||
|
Field[] fields = fieldCache.get(commandType);
|
||||||
|
if (fields == null) {
|
||||||
|
try {
|
||||||
|
fields = findFieldSequence(command, new ArrayDeque<Field>(2));
|
||||||
|
} catch (Throwable t) {
|
||||||
|
// Failed to get the field list
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fields == null) {
|
||||||
|
fields = FAILURE;
|
||||||
|
}
|
||||||
|
|
||||||
|
fieldCache.put(commandType, fields); // No need to use putIfAbsent()
|
||||||
|
}
|
||||||
|
|
||||||
|
if (fields == FAILURE) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
final int lastIndex = fields.length - 1;
|
||||||
|
for (int i = 0; i < lastIndex; i ++) {
|
||||||
|
command = (Runnable) get(fields[i], command);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (AbstractAioChannel) get(fields[lastIndex], command);
|
||||||
|
}
|
||||||
|
|
||||||
|
private Field[] findFieldSequence(Runnable command, Deque<Field> fields) throws Exception {
|
||||||
|
Class<?> commandType = command.getClass();
|
||||||
|
for (Field f: commandType.getDeclaredFields()) {
|
||||||
|
if (f.getType() == Runnable.class) {
|
||||||
|
f.setAccessible(true);
|
||||||
|
fields.addLast(f);
|
||||||
|
try {
|
||||||
|
Field[] ret = findFieldSequence((Runnable) get(f, command), fields);
|
||||||
|
if (ret != null) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
fields.removeLast();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (f.getType() == Object.class) {
|
||||||
|
f.setAccessible(true);
|
||||||
|
fields.addLast(f);
|
||||||
|
try {
|
||||||
|
Object candidate = get(f, command);
|
||||||
|
if (candidate instanceof AbstractAioChannel) {
|
||||||
|
return fields.toArray(new Field[fields.size()]);
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
fields.removeLast();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected Object get(Field f, Object command) throws Exception {
|
||||||
|
return f.get(command);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,28 @@
|
|||||||
|
package io.netty.channel.socket.aio;
|
||||||
|
|
||||||
|
import java.lang.reflect.Field;
|
||||||
|
|
||||||
|
import sun.misc.Unsafe;
|
||||||
|
|
||||||
|
@SuppressWarnings("restriction")
|
||||||
|
class UnsafeAioChannelFinder extends DefaultAioChannelFinder {
|
||||||
|
private static final Unsafe UNSAFE = getUnsafe();
|
||||||
|
|
||||||
|
@Override
|
||||||
|
protected Object get(Field f, Object command) throws Exception {
|
||||||
|
// using Unsafe to directly access the field. This should be
|
||||||
|
// faster then "pure" reflection
|
||||||
|
long offset = UNSAFE.objectFieldOffset(f);
|
||||||
|
return UNSAFE.getObject(command, offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
private static Unsafe getUnsafe() {
|
||||||
|
try {
|
||||||
|
Field singleoneInstanceField = Unsafe.class.getDeclaredField("theUnsafe");
|
||||||
|
singleoneInstanceField.setAccessible(true);
|
||||||
|
return (Unsafe) singleoneInstanceField.get(null);
|
||||||
|
} catch (Throwable cause) {
|
||||||
|
throw new RuntimeException("Error while obtaining sun.misc.Unsafe", cause);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user