Minimize memory footprint for AbstractChannelHandlerContext for handlers that execute in the EventExecutor. (#8786)
Motivation: We cache the Runnable for some tasks to reduce GC pressure in 4 different fields. This gives overhead in terms of memory usage in all cases, even if we always execute in the EventExecutor (which is the case most of the times). Modifications: Move the 4 fields to another class and only have one reference to this in AbstractChannelHandlerContext. This gives a small overhead in the case of execution that is done outside of the EventExecutor but reduce memory footprint in the more likily execution case. Result: Less memory used per AbstractChannelHandlerContext in most cases.
This commit is contained in:
parent
e941cbe27a
commit
185efa5b7c
@ -116,10 +116,7 @@ abstract class AbstractChannelHandlerContext extends DefaultAttributeMap
|
|||||||
|
|
||||||
// Lazily instantiated tasks used to trigger events to a handler with different executor.
|
// Lazily instantiated tasks used to trigger events to a handler with different executor.
|
||||||
// There is no need to make this volatile as at worse it will just create a few more instances then needed.
|
// There is no need to make this volatile as at worse it will just create a few more instances then needed.
|
||||||
private Runnable invokeChannelReadCompleteTask;
|
private Tasks invokeTasks;
|
||||||
private Runnable invokeReadTask;
|
|
||||||
private Runnable invokeChannelWritableStateChangedTask;
|
|
||||||
private Runnable invokeFlushTask;
|
|
||||||
|
|
||||||
private volatile int handlerState = INIT;
|
private volatile int handlerState = INIT;
|
||||||
|
|
||||||
@ -486,11 +483,11 @@ abstract class AbstractChannelHandlerContext extends DefaultAttributeMap
|
|||||||
if (executor.inEventLoop()) {
|
if (executor.inEventLoop()) {
|
||||||
next.invokeChannelReadComplete();
|
next.invokeChannelReadComplete();
|
||||||
} else {
|
} else {
|
||||||
Runnable task = next.invokeChannelReadCompleteTask;
|
Tasks tasks = next.invokeTasks;
|
||||||
if (task == null) {
|
if (tasks == null) {
|
||||||
next.invokeChannelReadCompleteTask = task = next::invokeChannelReadComplete;
|
next.invokeTasks = tasks = new Tasks(next);
|
||||||
}
|
}
|
||||||
executor.execute(task);
|
executor.execute(tasks.invokeChannelReadCompleteTask);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -517,11 +514,11 @@ abstract class AbstractChannelHandlerContext extends DefaultAttributeMap
|
|||||||
if (executor.inEventLoop()) {
|
if (executor.inEventLoop()) {
|
||||||
next.invokeChannelWritabilityChanged();
|
next.invokeChannelWritabilityChanged();
|
||||||
} else {
|
} else {
|
||||||
Runnable task = next.invokeChannelWritableStateChangedTask;
|
Tasks tasks = next.invokeTasks;
|
||||||
if (task == null) {
|
if (tasks == null) {
|
||||||
next.invokeChannelWritableStateChangedTask = task = next::invokeChannelWritabilityChanged;
|
next.invokeTasks = tasks = new Tasks(next);
|
||||||
}
|
}
|
||||||
executor.execute(task);
|
executor.execute(tasks.invokeChannelWritableStateChangedTask);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -781,11 +778,11 @@ abstract class AbstractChannelHandlerContext extends DefaultAttributeMap
|
|||||||
if (executor.inEventLoop()) {
|
if (executor.inEventLoop()) {
|
||||||
next.invokeRead();
|
next.invokeRead();
|
||||||
} else {
|
} else {
|
||||||
Runnable task = next.invokeReadTask;
|
Tasks tasks = next.invokeTasks;
|
||||||
if (task == null) {
|
if (tasks == null) {
|
||||||
next.invokeReadTask = task = next::invokeRead;
|
next.invokeTasks = tasks = new Tasks(next);
|
||||||
}
|
}
|
||||||
executor.execute(task);
|
executor.execute(tasks.invokeReadTask);
|
||||||
}
|
}
|
||||||
|
|
||||||
return this;
|
return this;
|
||||||
@ -838,11 +835,11 @@ abstract class AbstractChannelHandlerContext extends DefaultAttributeMap
|
|||||||
if (executor.inEventLoop()) {
|
if (executor.inEventLoop()) {
|
||||||
next.invokeFlush();
|
next.invokeFlush();
|
||||||
} else {
|
} else {
|
||||||
Runnable task = next.invokeFlushTask;
|
Tasks tasks = next.invokeTasks;
|
||||||
if (task == null) {
|
if (tasks == null) {
|
||||||
next.invokeFlushTask = task = next::invokeFlush;
|
next.invokeTasks = tasks = new Tasks(next);
|
||||||
}
|
}
|
||||||
safeExecute(executor, task, channel().voidPromise(), null);
|
safeExecute(executor, tasks.invokeFlushTask, channel().voidPromise(), null);
|
||||||
}
|
}
|
||||||
|
|
||||||
return this;
|
return this;
|
||||||
@ -1261,4 +1258,18 @@ abstract class AbstractChannelHandlerContext extends DefaultAttributeMap
|
|||||||
ctx.invokeFlush();
|
ctx.invokeFlush();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static final class Tasks {
|
||||||
|
private final Runnable invokeChannelReadCompleteTask;
|
||||||
|
private final Runnable invokeReadTask;
|
||||||
|
private final Runnable invokeChannelWritableStateChangedTask;
|
||||||
|
private final Runnable invokeFlushTask;
|
||||||
|
|
||||||
|
Tasks(AbstractChannelHandlerContext next) {
|
||||||
|
invokeChannelReadCompleteTask = next::invokeChannelReadComplete;
|
||||||
|
invokeReadTask = next::invokeRead;
|
||||||
|
invokeChannelWritableStateChangedTask = next::invokeChannelWritabilityChanged;
|
||||||
|
invokeFlushTask = next::invokeFlush;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user