30
30
import java .util .Set ;
31
31
import java .util .concurrent .ConcurrentHashMap ;
32
32
import java .util .concurrent .atomic .LongAdder ;
33
- import java .util .function .Supplier ;
34
33
import java .util .stream .Stream ;
35
34
import jdk .internal .access .JavaLangAccess ;
36
35
import jdk .internal .access .SharedSecrets ;
42
41
public class SharedThreadContainer extends ThreadContainer implements AutoCloseable {
43
42
private static final JavaLangAccess JLA = SharedSecrets .getJavaLangAccess ();
44
43
private static final VarHandle CLOSED ;
44
+ private static final VarHandle VIRTUAL_THREADS ;
45
45
static {
46
46
try {
47
47
MethodHandles .Lookup l = MethodHandles .lookup ();
48
- CLOSED = l .findVarHandle (SharedThreadContainer .class , "closed" , boolean .class );
48
+ CLOSED = l .findVarHandle (SharedThreadContainer .class ,
49
+ "closed" , boolean .class );
50
+ VIRTUAL_THREADS = l .findVarHandle (SharedThreadContainer .class ,
51
+ "virtualThreads" , Set .class );
49
52
} catch (Exception e ) {
50
53
throw new InternalError (e );
51
54
}
52
55
}
53
56
54
- private final ThreadContainer parent ;
57
+ // name of container, used by toString
55
58
private final String name ;
56
59
57
- // thread count, null if not tracking
60
+ // the number of threads in the container
58
61
private final LongAdder threadCount ;
59
62
60
- // the virtual threads in the container, null if not tracking
61
- private final Set <Thread > virtualThreads ;
62
-
63
- // supplier of threads, can be set lazily.
64
- private volatile Supplier <Stream <Thread >> threadsSupplier ;
63
+ // the virtual threads in the container, created lazily
64
+ private volatile Set <Thread > virtualThreads ;
65
65
66
+ // the key for this container in the registry
66
67
private volatile Object key ;
68
+
69
+ // set to true when the container is closed
67
70
private volatile boolean closed ;
68
71
69
72
/**
70
73
* Initialize a new SharedThreadContainer.
71
- * @param parent the (unowned) parent
72
74
* @param name the container name, can be null
73
- * @param trackThreads true to track threads
74
75
*/
75
- private SharedThreadContainer (ThreadContainer parent , String name , boolean trackThreads ) {
76
- super (true );
77
- this .parent = parent ;
76
+ private SharedThreadContainer (String name ) {
77
+ super (/*shared*/ true );
78
78
this .name = name ;
79
- if (trackThreads ) {
80
- this .threadCount = new LongAdder ();
81
- this .virtualThreads = ConcurrentHashMap .newKeySet ();
82
- } else {
83
- this .threadCount = null ;
84
- this .virtualThreads = null ;
85
- }
86
- }
87
-
88
- private static SharedThreadContainer create (ThreadContainer parent ,
89
- String name ,
90
- boolean trackThreads ) {
91
- var container = new SharedThreadContainer (parent , name , trackThreads );
92
- container .key = ThreadContainers .registerContainer (container );
93
- return container ;
79
+ this .threadCount = new LongAdder ();
94
80
}
95
81
96
82
/**
@@ -100,7 +86,10 @@ private static SharedThreadContainer create(ThreadContainer parent,
100
86
public static SharedThreadContainer create (ThreadContainer parent , String name ) {
101
87
if (parent .owner () != null )
102
88
throw new IllegalArgumentException ("parent has owner" );
103
- return create (parent , name , true );
89
+ var container = new SharedThreadContainer (name );
90
+ // register the container to allow discovery by serviceability tools
91
+ container .key = ThreadContainers .registerContainer (container );
92
+ return container ;
104
93
}
105
94
106
95
/**
@@ -111,74 +100,53 @@ public static SharedThreadContainer create(String name) {
111
100
return create (ThreadContainers .root (), name );
112
101
}
113
102
114
- /**
115
- * Creates a shared thread container with the given name. Its parent will be
116
- * the root thread container. The container optionally tracks threads.
117
- */
118
- public static SharedThreadContainer create (String name , boolean trackThreads ) {
119
- return create (ThreadContainers .root (), name , trackThreads );
120
- }
121
-
122
- @ Override
123
- public ThreadContainer parent () {
124
- return parent ;
125
- }
126
-
127
- @ Override
128
- public String name () {
129
- return name ;
130
- }
131
-
132
103
@ Override
133
104
public Thread owner () {
134
105
return null ;
135
106
}
136
107
137
108
@ Override
138
109
public void onStart (Thread thread ) {
139
- if (virtualThreads != null && thread .isVirtual ())
140
- virtualThreads .add (thread );
141
- if (threadCount != null )
142
- threadCount .add (1L );
110
+ // virtual threads needs to be tracked
111
+ if (thread .isVirtual ()) {
112
+ Set <Thread > vthreads = this .virtualThreads ;
113
+ if (vthreads == null ) {
114
+ vthreads = ConcurrentHashMap .newKeySet ();
115
+ if (!VIRTUAL_THREADS .compareAndSet (this , null , vthreads )) {
116
+ // lost the race
117
+ vthreads = this .virtualThreads ;
118
+ }
119
+ }
120
+ vthreads .add (thread );
121
+ }
122
+ threadCount .add (1L );
143
123
}
144
124
145
125
@ Override
146
126
public void onExit (Thread thread ) {
147
- if (threadCount != null )
148
- threadCount .add (-1L );
149
- if (virtualThreads != null && thread .isVirtual ())
127
+ threadCount .add (-1L );
128
+ if (thread .isVirtual ())
150
129
virtualThreads .remove (thread );
151
130
}
152
131
153
132
@ Override
154
133
public long threadCount () {
155
- if (threadCount != null ) {
156
- return threadCount .sum ();
157
- } else {
158
- return threads ().mapToLong (e -> 1L ).sum ();
159
- }
160
- }
161
-
162
- /**
163
- * Sets the object that enumerates the threads in the container.
164
- */
165
- public void threadsSupplier (Supplier <Stream <Thread >> threadsSupplier ) {
166
- this .threadsSupplier = Objects .requireNonNull (threadsSupplier );
134
+ return threadCount .sum ();
167
135
}
168
136
169
137
@ Override
170
138
public Stream <Thread > threads () {
171
- Supplier <Stream <Thread >> threadsSupplier = this .threadsSupplier ;
172
- if (threadsSupplier != null ) {
173
- return threadsSupplier .get ();
139
+ // live platform threads in this container
140
+ Stream <Thread > platformThreads = Stream .of (JLA .getAllThreads ())
141
+ .filter (t -> JLA .threadContainer (t ) == this );
142
+ Set <Thread > vthreads = this .virtualThreads ;
143
+ if (vthreads == null ) {
144
+ // live platform threads only, no virtual threads
145
+ return platformThreads ;
174
146
} else {
175
- Stream <Thread > platformThreads = Stream .of (JLA .getAllThreads ())
176
- .filter (t -> JLA .threadContainer (t ) == this );
177
- if (virtualThreads == null ) {
178
- return platformThreads ;
179
- } else {
180
- return Stream .concat (platformThreads , virtualThreads .stream ());
181
- }
147
+ // all live threads in this container
148
+ return Stream .concat (platformThreads ,
149
+ vthreads .stream ().filter (Thread ::isAlive ));
182
150
}
183
151
}
184
152
@@ -207,7 +175,6 @@ public void close() {
207
175
@ Override
208
176
public String toString () {
209
177
String id = Objects .toIdentityString (this );
210
- String name = name ();
211
178
if (name != null ) {
212
179
return name + "/" + id ;
213
180
} else {
0 commit comments