Skip to content

Commit 9a717da

Browse files
authored
Add thread-core-ratio config option and change default to 1 (#11666)
* Add thread-core-ratio config option and change default to 1 We already have options to set the number of threads for event loops explicitly, but none that scale with host core count. This PR adds a thread-core-ratio config option that allows for dynamically changing event loop thread count based on core count. Netty previously did the same thing when num-threads is 0, but this PR moves that logic to our code. The default ratio is changed from nettys 2 to 1. A single thread per core reduces jitter and should achieve the same throughput in most scenarios. * fix test * fix test
1 parent 3709b7e commit 9a717da

File tree

7 files changed

+101
-7
lines changed

7 files changed

+101
-7
lines changed

http-netty/src/main/java/io/micronaut/http/netty/channel/DefaultEventLoopGroupConfiguration.java

+24
Original file line numberDiff line numberDiff line change
@@ -36,6 +36,7 @@
3636
public class DefaultEventLoopGroupConfiguration implements EventLoopGroupConfiguration {
3737

3838
private final int numThreads;
39+
private final double threadCoreRatio;
3940
private final Integer ioRatio;
4041
private final boolean preferNativeTransport;
4142
private final String name;
@@ -48,6 +49,8 @@ public class DefaultEventLoopGroupConfiguration implements EventLoopGroupConfigu
4849
*
4950
* @param name The name of the group
5051
* @param numThreads The number of threads
52+
* @param threadCoreRatio The number of threads per core to use if
53+
* {@link #getNumThreads()} is set to 0.
5154
* @param ioRatio The IO ratio (optional)
5255
* @param preferNativeTransport Whether native transport is to be preferred
5356
* @param executor A named executor service to use for event loop threads
@@ -62,6 +65,7 @@ public class DefaultEventLoopGroupConfiguration implements EventLoopGroupConfigu
6265
public DefaultEventLoopGroupConfiguration(
6366
@Parameter String name,
6467
@Bindable(defaultValue = "0") int numThreads,
68+
@Bindable(defaultValue = DEFAULT_THREAD_CORE_RATIO + "") double threadCoreRatio,
6569
@Nullable Integer ioRatio,
6670
@Bindable(defaultValue = StringUtils.FALSE) boolean preferNativeTransport,
6771
@Nullable String executor,
@@ -70,6 +74,7 @@ public DefaultEventLoopGroupConfiguration(
7074
) {
7175
this.name = name;
7276
this.numThreads = numThreads;
77+
this.threadCoreRatio = threadCoreRatio;
7378
this.ioRatio = ioRatio;
7479
this.preferNativeTransport = preferNativeTransport;
7580
this.executor = executor;
@@ -79,12 +84,26 @@ public DefaultEventLoopGroupConfiguration(
7984
.orElse(Duration.ofSeconds(DEFAULT_SHUTDOWN_TIMEOUT));
8085
}
8186

87+
@Deprecated
88+
public DefaultEventLoopGroupConfiguration(
89+
String name,
90+
int numThreads,
91+
Integer ioRatio,
92+
boolean preferNativeTransport,
93+
String executor,
94+
Duration shutdownQuietPeriod,
95+
Duration shutdownTimeout
96+
) {
97+
this(name, numThreads, DEFAULT_THREAD_CORE_RATIO, ioRatio, preferNativeTransport, executor, shutdownQuietPeriod, shutdownTimeout);
98+
}
99+
82100
/**
83101
* Default constructor.
84102
*/
85103
public DefaultEventLoopGroupConfiguration() {
86104
this.name = DEFAULT;
87105
this.numThreads = 0;
106+
this.threadCoreRatio = DEFAULT_THREAD_CORE_RATIO;
88107
this.ioRatio = null;
89108
this.preferNativeTransport = false;
90109
this.executor = null;
@@ -100,6 +119,11 @@ public int getNumThreads() {
100119
return numThreads;
101120
}
102121

122+
@Override
123+
public double getThreadCoreRatio() {
124+
return threadCoreRatio;
125+
}
126+
103127
/**
104128
* @return The I/O ratio.
105129
*/

http-netty/src/main/java/io/micronaut/http/netty/channel/DefaultEventLoopGroupRegistry.java

+17-1
Original file line numberDiff line numberDiff line change
@@ -29,6 +29,7 @@
2929
import io.micronaut.inject.qualifiers.Qualifiers;
3030
import io.netty.channel.EventLoopGroup;
3131
import io.netty.channel.nio.NioEventLoopGroup;
32+
import io.netty.util.NettyRuntime;
3233
import io.netty.util.concurrent.DefaultThreadFactory;
3334
import jakarta.annotation.PreDestroy;
3435
import jakarta.inject.Named;
@@ -105,7 +106,7 @@ protected EventLoopGroup eventLoopGroup(EventLoopGroupConfiguration configuratio
105106
if (executor != null) {
106107
eventLoopGroup = beanLocator.findBean(Executor.class, Qualifiers.byName(executor))
107108
.map(executorService -> eventLoopGroupFactory.createEventLoopGroup(
108-
configuration.getNumThreads(),
109+
numThreads(configuration),
109110
executorService,
110111
configuration.getIoRatio().orElse(null)
111112
)).orElseThrow(() -> new ConfigurationException("No executor service configured for name: " + executor));
@@ -157,4 +158,19 @@ public Optional<EventLoopGroupConfiguration> getEventLoopGroupConfiguration(@Non
157158
ArgumentUtils.requireNonNull("name", name);
158159
return beanLocator.findBean(EventLoopGroupConfiguration.class, Qualifiers.byName(name));
159160
}
161+
162+
/**
163+
* Calculate the number of threads from {@link EventLoopGroupConfiguration#getNumThreads()} and
164+
* {@link EventLoopGroupConfiguration#getThreadCoreRatio()}.
165+
*
166+
* @param configuration The configuration
167+
* @return The actual number of threads to use
168+
*/
169+
public static int numThreads(EventLoopGroupConfiguration configuration) {
170+
int explicit = configuration.getNumThreads();
171+
if (explicit != 0) {
172+
return explicit;
173+
}
174+
return Math.toIntExact(Math.round(configuration.getThreadCoreRatio() * NettyRuntime.availableProcessors()));
175+
}
160176
}

http-netty/src/main/java/io/micronaut/http/netty/channel/EventLoopGroupConfiguration.java

+15-1
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,10 @@ public interface EventLoopGroupConfiguration extends Named {
3939
* The default.
4040
*/
4141
String DEFAULT_LOOP = EVENT_LOOPS + "." + DEFAULT;
42+
/**
43+
* Default {@link #getThreadCoreRatio()}.
44+
*/
45+
double DEFAULT_THREAD_CORE_RATIO = 1.0;
4246

4347
/**
4448
* The default shutdown quiet period in seconds.
@@ -55,10 +59,20 @@ public interface EventLoopGroupConfiguration extends Named {
5559
long DEFAULT_SHUTDOWN_TIMEOUT = 15;
5660

5761
/**
58-
* @return The number of threads for the event loop
62+
* @return The number of threads for the event loop, or 0 to use {@link #getThreadCoreRatio()}
5963
*/
6064
int getNumThreads();
6165

66+
/**
67+
* The number of threads per core to use if {@link #getNumThreads()} is set to 0.
68+
*
69+
* @return The thread-to-core ratio
70+
* @since 4.8.0
71+
*/
72+
default double getThreadCoreRatio() {
73+
return DEFAULT_THREAD_CORE_RATIO;
74+
}
75+
6276
/**
6377
* @return The I/O ratio.
6478
*/

http-netty/src/main/java/io/micronaut/http/netty/channel/EventLoopGroupFactory.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -73,7 +73,7 @@ default EventLoopGroup createEventLoopGroup(
7373
ArgumentUtils.requireNonNull("configuration", configuration);
7474
ArgumentUtils.requireNonNull("threadFactory", threadFactory);
7575
return createEventLoopGroup(
76-
configuration.getNumThreads(),
76+
DefaultEventLoopGroupRegistry.numThreads(configuration),
7777
threadFactory,
7878
configuration.getIoRatio().orElse(null)
7979
);

http-netty/src/test/groovy/io/micronaut/http/netty/channel/EventLoopGroupSpec.groovy

+25-2
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,7 @@ class EventLoopGroupSpec extends Specification {
3636

3737
then:
3838
!eventLoopGroup.isTerminated()
39-
eventLoopGroup.executorCount() == NettyRuntime.availableProcessors() * 2
39+
eventLoopGroup.executorCount() == NettyRuntime.availableProcessors()
4040
ResourceLeakDetector.level == ResourceLeakDetector.Level.DISABLED
4141

4242
when:
@@ -89,6 +89,29 @@ class EventLoopGroupSpec extends Specification {
8989
eventLoopGroup.isShuttingDown()
9090
}
9191

92+
void "test core ratio"(int ratio) {
93+
given:
94+
def context = ApplicationContext.run(
95+
'micronaut.netty.event-loops.default.thread-core-ratio': ratio
96+
)
97+
98+
when:
99+
def eventLoopGroup = context.getBean(EventLoopGroup)
100+
101+
then:
102+
!eventLoopGroup.isTerminated()
103+
eventLoopGroup.executorCount() == ratio * Runtime.getRuntime().availableProcessors()
104+
105+
when:
106+
context.close()
107+
108+
then:
109+
eventLoopGroup.isShuttingDown()
110+
111+
where:
112+
ratio << [1, 2, 3]
113+
}
114+
92115
void "test configure additional event loop groups"() {
93116
given:
94117
def context = ApplicationContext.run(
@@ -106,7 +129,7 @@ class EventLoopGroupSpec extends Specification {
106129

107130
then:
108131
!eventLoopGroup.isTerminated()
109-
eventLoopGroup.executorCount() == NettyRuntime.availableProcessors() * 2
132+
eventLoopGroup.executorCount() == NettyRuntime.availableProcessors()
110133

111134
when:
112135
def eventLoopGroup2 = context.getBean(EventLoopGroup, Qualifiers.byName("one"))

http-server-netty/src/main/java/io/micronaut/http/server/netty/NettyHttpServer.java

+3-2
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,7 @@
3232
import io.micronaut.http.context.event.HttpRequestTerminatedEvent;
3333
import io.micronaut.http.netty.channel.ChannelPipelineListener;
3434
import io.micronaut.http.netty.channel.DefaultEventLoopGroupConfiguration;
35+
import io.micronaut.http.netty.channel.DefaultEventLoopGroupRegistry;
3536
import io.micronaut.http.netty.channel.EventLoopGroupConfiguration;
3637
import io.micronaut.http.netty.channel.NettyChannelType;
3738
import io.micronaut.http.netty.channel.converters.ChannelOptionFactory;
@@ -770,7 +771,7 @@ private EventLoopGroup newEventLoopGroup(EventLoopGroupConfiguration config) {
770771
.flatMap(name -> applicationContext.findBean(ExecutorService.class, Qualifiers.byName(name))).orElse(null);
771772
if (executorService != null) {
772773
return nettyEmbeddedServices.createEventLoopGroup(
773-
config.getNumThreads(),
774+
DefaultEventLoopGroupRegistry.numThreads(config),
774775
executorService,
775776
config.getIoRatio().orElse(null)
776777
);
@@ -964,7 +965,7 @@ protected void initChannel(Channel ch) throws Exception {
964965
}
965966
}
966967

967-
private static class DomainSocketHolder {
968+
private static final class DomainSocketHolder {
968969
@NonNull
969970
private static SocketAddress makeDomainSocketAddress(String path) {
970971
try {

http-server-netty/src/main/java/io/micronaut/http/server/netty/configuration/NettyHttpServerConfiguration.java

+16
Original file line numberDiff line numberDiff line change
@@ -1297,6 +1297,7 @@ public boolean getPublic() {
12971297
*/
12981298
public abstract static class EventLoopConfig implements EventLoopGroupConfiguration {
12991299
private int threads;
1300+
private double threadCoreRatio = DEFAULT_THREAD_CORE_RATIO;
13001301
private Integer ioRatio;
13011302
private String executor;
13021303
private boolean preferNativeTransport = false;
@@ -1415,6 +1416,21 @@ public int getNumThreads() {
14151416
return threads;
14161417
}
14171418

1419+
@Override
1420+
public double getThreadCoreRatio() {
1421+
return threadCoreRatio;
1422+
}
1423+
1424+
/**
1425+
* The number of threads per core to use if {@link #getNumThreads()} is set to 0.
1426+
*
1427+
* @param threadCoreRatio The thread-to-core ratio
1428+
* @since 4.8.0
1429+
*/
1430+
public void setThreadCoreRatio(double threadCoreRatio) {
1431+
this.threadCoreRatio = threadCoreRatio;
1432+
}
1433+
14181434
@Override
14191435
public boolean isPreferNativeTransport() {
14201436
return preferNativeTransport;

0 commit comments

Comments
 (0)