Skip to content

Commit 20f6e42

Browse files
committedMay 1, 2022
Cleanup, removed unused methods, address review comments
1 parent 896a731 commit 20f6e42

File tree

8 files changed

+119
-142
lines changed

8 files changed

+119
-142
lines changed
 

‎src/java.base/share/classes/java/lang/Thread.java

+5-4
Original file line numberDiff line numberDiff line change
@@ -1543,15 +1543,16 @@ void start(ThreadContainer container) {
15431543
if (holder.threadStatus != 0)
15441544
throw new IllegalThreadStateException();
15451545

1546+
// bind thread to container
1547+
setThreadContainer(container);
1548+
1549+
// start thread
15461550
boolean started = false;
15471551
container.onStart(this); // may throw
15481552
try {
15491553
// extent locals may be inherited
15501554
inheritExtentLocalBindings(container);
15511555

1552-
// bind thread to container
1553-
setThreadContainer(container);
1554-
15551556
start0();
15561557
started = true;
15571558
} finally {
@@ -3063,7 +3064,7 @@ static ThreadGroup virtualThreadGroup() {
30633064
int threadLocalRandomSecondarySeed;
30643065

30653066
/** The thread container that this thread is in */
3066-
@Stable private ThreadContainer container;
3067+
private volatile ThreadContainer container; // @Stable candidate?
30673068
ThreadContainer threadContainer() {
30683069
return container;
30693070
}

‎src/java.base/share/classes/java/lang/VirtualThread.java

+10-3
Original file line numberDiff line numberDiff line change
@@ -448,15 +448,16 @@ void start(ThreadContainer container) {
448448
throw new IllegalThreadStateException("Already started");
449449
}
450450

451+
// bind thread to container
452+
setThreadContainer(container);
453+
454+
// start thread
451455
boolean started = false;
452456
container.onStart(this); // may throw
453457
try {
454458
// extent locals may be inherited
455459
inheritExtentLocalBindings(container);
456460

457-
// bind thread to container
458-
setThreadContainer(container);
459-
460461
// submit task to run thread
461462
submitRunContinuation();
462463
started = true;
@@ -821,6 +822,12 @@ Thread.State threadState() {
821822
case NEW:
822823
return Thread.State.NEW;
823824
case STARTED:
825+
// return NEW if thread container not yet set
826+
if (threadContainer() == null) {
827+
return Thread.State.NEW;
828+
} else {
829+
return Thread.State.RUNNABLE;
830+
}
824831
case RUNNABLE:
825832
case RUNNABLE_SUSPENDED:
826833
// runnable, not mounted

‎src/java.base/share/classes/java/util/concurrent/ThreadPerTaskExecutor.java

+25-17
Original file line numberDiff line numberDiff line change
@@ -36,13 +36,17 @@
3636
import java.util.concurrent.locks.LockSupport;
3737
import java.util.stream.Stream;
3838
import static java.util.concurrent.TimeUnit.NANOSECONDS;
39-
import jdk.internal.vm.SharedThreadContainer;
39+
import jdk.internal.access.JavaLangAccess;
40+
import jdk.internal.access.SharedSecrets;
41+
import jdk.internal.vm.ThreadContainer;
42+
import jdk.internal.vm.ThreadContainers;
4043

4144
/**
4245
* An ExecutorService that starts a new thread for each task. The number of
4346
* threads is unbounded.
4447
*/
45-
class ThreadPerTaskExecutor implements ExecutorService {
48+
class ThreadPerTaskExecutor extends ThreadContainer implements ExecutorService {
49+
private static final JavaLangAccess JLA = SharedSecrets.getJavaLangAccess();
4650
private static final Permission MODIFY_THREAD = new RuntimePermission("modifyThread");
4751
private static final VarHandle STATE;
4852
static {
@@ -54,34 +58,32 @@ class ThreadPerTaskExecutor implements ExecutorService {
5458
}
5559
}
5660

61+
private final ThreadFactory factory;
5762
private final Set<Thread> threads = ConcurrentHashMap.newKeySet();
5863
private final CountDownLatch terminationSignal = new CountDownLatch(1);
5964

60-
private final ThreadFactory factory;
61-
private final SharedThreadContainer container;
62-
6365
// states: RUNNING -> SHUTDOWN -> TERMINATED
6466
private static final int RUNNING = 0;
6567
private static final int SHUTDOWN = 1;
6668
private static final int TERMINATED = 2;
6769
private volatile int state;
6870

71+
// the key for this container in the registry
72+
private volatile Object key;
73+
6974
private ThreadPerTaskExecutor(ThreadFactory factory) {
75+
super(/*shared*/ true);
7076
this.factory = Objects.requireNonNull(factory);
71-
String name = Objects.toIdentityString(this);
72-
this.container = SharedThreadContainer.create(name, /*trackThreads*/ false);
73-
}
74-
75-
private ThreadPerTaskExecutor setThreadsSupplier() {
76-
container.threadsSupplier(this::threads);
77-
return this;
7877
}
7978

8079
/**
8180
* Creates a thread-per-task executor that creates threads using the given factory.
8281
*/
8382
static ThreadPerTaskExecutor create(ThreadFactory factory) {
84-
return new ThreadPerTaskExecutor(factory).setThreadsSupplier();
83+
var executor = new ThreadPerTaskExecutor(factory);
84+
// register it to allow discovery by serviceability tools
85+
executor.key = ThreadContainers.registerContainer(executor);
86+
return executor;
8587
}
8688

8789
/**
@@ -119,7 +121,7 @@ private void tryTerminate() {
119121
terminationSignal.countDown();
120122

121123
// remove from registry
122-
container.close();
124+
ThreadContainers.deregisterContainer(key);
123125
}
124126
}
125127

@@ -135,8 +137,14 @@ private void tryShutdownAndTerminate(boolean interruptThreads) {
135137
}
136138
}
137139

138-
private Stream<Thread> threads() {
139-
return threads.stream();
140+
@Override
141+
public Stream<Thread> threads() {
142+
return threads.stream().filter(Thread::isAlive);
143+
}
144+
145+
@Override
146+
public long threadCount() {
147+
return threads.size();
140148
}
141149

142150
@Override
@@ -238,7 +246,7 @@ private void start(Thread thread) {
238246
boolean started = false;
239247
try {
240248
if (state == RUNNING) {
241-
container.start(thread);
249+
JLA.start(thread, this);
242250
started = true;
243251
}
244252
} finally {

‎src/java.base/share/classes/jdk/internal/misc/ThreadFlock.java

+3-6
Original file line numberDiff line numberDiff line change
@@ -453,15 +453,15 @@ public boolean isClosed() {
453453
}
454454

455455
/**
456-
* {@return a stream of the threads in this flock}
456+
* {@return a stream of the live threads in this flock}
457457
* The elements of the stream are threads that were started in this flock
458458
* but have not terminated. The stream will reflect the set of threads in the
459459
* flock at some point at or since the creation of the stream. It may or may
460460
* not reflect changes to the set of threads subsequent to creation of the
461461
* stream.
462462
*/
463463
public Stream<Thread> threads() {
464-
return threads.stream();
464+
return threads.stream().filter(Thread::isAlive);
465465
}
466466

467467
/**
@@ -508,6 +508,7 @@ private static class ThreadContainerImpl extends ThreadContainer {
508508
private boolean closing;
509509

510510
ThreadContainerImpl(ThreadFlock flock) {
511+
super(/*shared*/ false);
511512
this.flock = flock;
512513
}
513514

@@ -563,10 +564,6 @@ protected boolean tryClose() {
563564
}
564565
}
565566

566-
@Override
567-
public String name() {
568-
return flock.name();
569-
}
570567
@Override
571568
public long threadCount() {
572569
return flock.threadCount();

‎src/java.base/share/classes/jdk/internal/vm/ExtentLocalContainer.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -85,7 +85,7 @@ public record BindingsSnapshot(Object extentLocalBindings,
8585
ExtentLocalContainer container) { }
8686

8787
/**
88-
* Returns a extent local bindings for the current thread.
88+
* Returns the extent local bindings for the current thread.
8989
*/
9090
public static BindingsSnapshot captureBindings() {
9191
return new BindingsSnapshot(JLA.extentLocalBindings(), latest());

‎src/java.base/share/classes/jdk/internal/vm/SharedThreadContainer.java

+45-78
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,6 @@
3030
import java.util.Set;
3131
import java.util.concurrent.ConcurrentHashMap;
3232
import java.util.concurrent.atomic.LongAdder;
33-
import java.util.function.Supplier;
3433
import java.util.stream.Stream;
3534
import jdk.internal.access.JavaLangAccess;
3635
import jdk.internal.access.SharedSecrets;
@@ -42,55 +41,42 @@
4241
public class SharedThreadContainer extends ThreadContainer implements AutoCloseable {
4342
private static final JavaLangAccess JLA = SharedSecrets.getJavaLangAccess();
4443
private static final VarHandle CLOSED;
44+
private static final VarHandle VIRTUAL_THREADS;
4545
static {
4646
try {
4747
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);
4952
} catch (Exception e) {
5053
throw new InternalError(e);
5154
}
5255
}
5356

54-
private final ThreadContainer parent;
57+
// name of container, used by toString
5558
private final String name;
5659

57-
// thread count, null if not tracking
60+
// the number of threads in the container
5861
private final LongAdder threadCount;
5962

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;
6565

66+
// the key for this container in the registry
6667
private volatile Object key;
68+
69+
// set to true when the container is closed
6770
private volatile boolean closed;
6871

6972
/**
7073
* Initialize a new SharedThreadContainer.
71-
* @param parent the (unowned) parent
7274
* @param name the container name, can be null
73-
* @param trackThreads true to track threads
7475
*/
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);
7878
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();
9480
}
9581

9682
/**
@@ -100,7 +86,10 @@ private static SharedThreadContainer create(ThreadContainer parent,
10086
public static SharedThreadContainer create(ThreadContainer parent, String name) {
10187
if (parent.owner() != null)
10288
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;
10493
}
10594

10695
/**
@@ -111,74 +100,53 @@ public static SharedThreadContainer create(String name) {
111100
return create(ThreadContainers.root(), name);
112101
}
113102

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-
132103
@Override
133104
public Thread owner() {
134105
return null;
135106
}
136107

137108
@Override
138109
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);
143123
}
144124

145125
@Override
146126
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())
150129
virtualThreads.remove(thread);
151130
}
152131

153132
@Override
154133
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();
167135
}
168136

169137
@Override
170138
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;
174146
} 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));
182150
}
183151
}
184152

@@ -207,7 +175,6 @@ public void close() {
207175
@Override
208176
public String toString() {
209177
String id = Objects.toIdentityString(this);
210-
String name = name();
211178
if (name != null) {
212179
return name + "/" + id;
213180
} else {

‎src/java.base/share/classes/jdk/internal/vm/ThreadContainer.java

+17-18
Original file line numberDiff line numberDiff line change
@@ -37,22 +37,10 @@ public abstract class ThreadContainer extends StackableScope {
3737
* @param shared true for a shared container, false for a container
3838
* owned by the current thread
3939
*/
40-
ThreadContainer(boolean shared) {
40+
protected ThreadContainer(boolean shared) {
4141
super(shared);
4242
}
4343

44-
/**
45-
* Creates a ThreadContainer owned by the current thread.
46-
*/
47-
protected ThreadContainer() {
48-
super(false);
49-
}
50-
51-
/**
52-
* Return the container name, null if not named.
53-
*/
54-
public abstract String name();
55-
5644
/**
5745
* Returns the parent of this container or null if this is the root container.
5846
*/
@@ -75,19 +63,30 @@ public long threadCount() {
7563
}
7664

7765
/**
78-
* Returns a stream of the threads in this container.
66+
* Returns a stream of the live threads in this container.
7967
*/
8068
public abstract Stream<Thread> threads();
8169

8270
/**
83-
* Invoked when a thread is started in the container
71+
* Invoked by Thread::start before the given Thread is started.
8472
*/
85-
public abstract void onStart(Thread thread);
73+
public void onStart(Thread thread) {
74+
// do nothing
75+
}
8676

8777
/**
88-
* Invoked when thread in container terminates.
78+
* Invoked when a Thread terminates or starting it fails.
79+
*
80+
* For a platform thread, this method is invoked by the thread itself when it
81+
* terminates. For a virtual thread, this method is invoked on its carrier
82+
* after the virtual thread has terminated.
83+
*
84+
* If starting the Thread failed then this method is invoked on the thread
85+
* that invoked onStart.
8986
*/
90-
public abstract void onExit(Thread thread);
87+
public void onExit(Thread thread) {
88+
// do nothing
89+
}
9190

9291
/**
9392
* The extent locals captured when the thread container was created.

‎src/java.base/share/classes/jdk/internal/vm/ThreadContainers.java

+13-15
Original file line numberDiff line numberDiff line change
@@ -101,10 +101,10 @@ static ThreadContainer parent(ThreadContainer container) {
101101
ThreadContainer parent = container.enclosingScope(ThreadContainer.class);
102102
if (parent != null)
103103
return parent;
104-
if ((parent = ThreadContainers.container(owner)) != null)
104+
if ((parent = container(owner)) != null)
105105
return parent;
106106
}
107-
ThreadContainer root = ThreadContainers.root();
107+
ThreadContainer root = root();
108108
return (container != root) ? root : null;
109109
}
110110

@@ -136,10 +136,16 @@ static Stream<ThreadContainer> children(ThreadContainer container) {
136136
/**
137137
* Returns the thread container that the given Thread is in or the root
138138
* container if not started in a container.
139+
* @throws IllegalStateException if the thread has not been started
139140
*/
140141
public static ThreadContainer container(Thread thread) {
141-
ThreadContainer container = JLA.threadContainer(thread);
142-
return (container != null) ? container : root();
142+
// thread container is set when the thread is started
143+
if (thread.isAlive() || thread.getState() == Thread.State.TERMINATED) {
144+
ThreadContainer container = JLA.threadContainer(thread);
145+
return (container != null) ? container : root();
146+
} else {
147+
throw new IllegalStateException("Thread not started");
148+
}
143149
}
144150

145151
/**
@@ -195,22 +201,13 @@ protected RootContainer() {
195201
super(true);
196202
}
197203
@Override
198-
public String name() {
199-
return "<root>";
200-
}
201-
@Override
202-
public Thread owner() {
203-
return null;
204-
}
205-
@Override
206204
public ThreadContainer parent() {
207205
return null;
208206
}
209207
@Override
210208
public String toString() {
211-
return name();
209+
return "<root>";
212210
}
213-
214211
@Override
215212
public StackableScope previous() {
216213
return null;
@@ -246,7 +243,8 @@ public long threadCount() {
246243
}
247244
@Override
248245
public Stream<Thread> threads() {
249-
return Stream.concat(platformThreads(), VTHREADS.stream());
246+
return Stream.concat(platformThreads(),
247+
VTHREADS.stream().filter(Thread::isAlive));
250248
}
251249
}
252250

0 commit comments

Comments
 (0)
Please sign in to comment.