Extract ChannelFinder into separate top level classes

This commit is contained in:
Trustin Lee 2012-08-28 16:14:35 +09:00
parent e55a1f11b5
commit fe4a8d6dc9
4 changed files with 121 additions and 115 deletions

View File

@ -0,0 +1,5 @@
package io.netty.channel.socket.aio;
interface AioChannelFinder {
AbstractAioChannel findChannel(Runnable command) throws Exception;
}

View File

@ -15,47 +15,39 @@
*/
package io.netty.channel.socket.aio;
import io.netty.channel.ChannelTaskScheduler;
import io.netty.channel.EventExecutor;
import io.netty.channel.EventLoopException;
import io.netty.channel.MultithreadEventLoopGroup;
import io.netty.channel.ChannelTaskScheduler;
import io.netty.logging.InternalLogger;
import io.netty.logging.InternalLoggerFactory;
import io.netty.util.internal.DetectionUtil;
import java.io.IOException;
import java.lang.reflect.Field;
import java.nio.channels.AsynchronousChannelGroup;
import java.util.ArrayDeque;
import java.util.Collections;
import java.util.Deque;
import java.util.List;
import java.util.concurrent.AbstractExecutorService;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.ThreadFactory;
import java.util.concurrent.TimeUnit;
import sun.misc.Unsafe;
public class AioEventLoopGroup extends MultithreadEventLoopGroup {
private static final InternalLogger LOGGER = InternalLoggerFactory.getInstance(AioEventLoopGroup.class);
private static final ChannelFinder CHANNEL_FINDER;
private static final AioChannelFinder CHANNEL_FINDER;
static {
ChannelFinder finder;
AioChannelFinder finder;
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()) {
finder = new UnsafeChannelFinder();
finder = new UnsafeAioChannelFinder();
} else {
finder = new ReflectionChannelFinder();
finder = new DefaultAioChannelFinder();
}
} catch (Throwable t) {
LOGGER.debug("Unable to instance UnsafeChannelFinder fallback to ReflectionChannelFinder", t);
finder = new ReflectionChannelFinder();
LOGGER.debug(String.format(
"Unable to instance the optimal %s implementation - falling back to %s.",
AioChannelFinder.class.getSimpleName(), DefaultAioChannelFinder.class.getSimpleName()), t);
finder = new DefaultAioChannelFinder();
}
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);
}
}
}
}

View File

@ -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);
}
}

View File

@ -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);
}
}
}