1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 package io.netty.util.concurrent;
17
18 import java.util.concurrent.atomic.AtomicInteger;
19 import java.util.concurrent.atomic.AtomicLong;
20
21
22
23
24 public final class DefaultEventExecutorChooserFactory implements EventExecutorChooserFactory {
25
26 public static final DefaultEventExecutorChooserFactory INSTANCE = new DefaultEventExecutorChooserFactory();
27
28 private DefaultEventExecutorChooserFactory() { }
29
30 @Override
31 public EventExecutorChooser newChooser(EventExecutor[] executors) {
32 if (isPowerOfTwo(executors.length)) {
33 return new PowerOfTwoEventExecutorChooser(executors);
34 } else {
35 return new GenericEventExecutorChooser(executors);
36 }
37 }
38
39 private static boolean isPowerOfTwo(int val) {
40 return (val & -val) == val;
41 }
42
43 private static final class PowerOfTwoEventExecutorChooser implements EventExecutorChooser {
44 private final AtomicInteger idx = new AtomicInteger();
45 private final EventExecutor[] executors;
46
47 PowerOfTwoEventExecutorChooser(EventExecutor[] executors) {
48 this.executors = executors;
49 }
50
51 @Override
52 public EventExecutor next() {
53 return executors[idx.getAndIncrement() & executors.length - 1];
54 }
55 }
56
57 private static final class GenericEventExecutorChooser implements EventExecutorChooser {
58
59
60
61 private final AtomicLong idx = new AtomicLong();
62 private final EventExecutor[] executors;
63
64 GenericEventExecutorChooser(EventExecutor[] executors) {
65 this.executors = executors;
66 }
67
68 @Override
69 public EventExecutor next() {
70 return executors[(int) Math.abs(idx.getAndIncrement() % executors.length)];
71 }
72 }
73 }