Skip to content

Commit 3e509c8

Browse files
author
Doug Lea
committedNov 26, 2024
8344773: SM cleanup in ForkJoinPool
Reviewed-by: alanb
1 parent 6da3ecd commit 3e509c8

File tree

2 files changed

+49
-131
lines changed

2 files changed

+49
-131
lines changed
 

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

+24-106
Original file line numberDiff line numberDiff line change
@@ -37,12 +37,6 @@
3737

3838
import java.lang.Thread.UncaughtExceptionHandler;
3939
import java.lang.reflect.Field;
40-
import java.security.AccessController;
41-
import java.security.AccessControlContext;
42-
import java.security.Permission;
43-
import java.security.Permissions;
44-
import java.security.PrivilegedAction;
45-
import java.security.ProtectionDomain;
4640
import java.util.ArrayList;
4741
import java.util.Collection;
4842
import java.util.Collections;
@@ -811,9 +805,7 @@ public class ForkJoinPool extends AbstractExecutorService {
811805
* initialization. Since it (or any other created pool) need
812806
* never be used, we minimize initial construction overhead and
813807
* footprint to the setup of about a dozen fields, although with
814-
* some System property parsing and security processing that takes
815-
* far longer than the actual construction when SecurityManagers
816-
* are used or properties are set. The common pool is
808+
* some System property parsing properties are set. The common pool is
817809
* distinguished by having a null workerNamePrefix (which is an
818810
* odd convention, but avoids the need to decode status in factory
819811
* classes). It also has PRESET_SIZE config set if parallelism
@@ -839,13 +831,12 @@ public class ForkJoinPool extends AbstractExecutorService {
839831
*
840832
* As a more appropriate default in managed environments, unless
841833
* overridden by system properties, we use workers of subclass
842-
* InnocuousForkJoinWorkerThread when there is a SecurityManager
843-
* present. These workers have no permissions set, do not belong
844-
* to any user-defined ThreadGroup, and clear all ThreadLocals and
845-
* reset the ContextClassLoader before (re)activating to execute
846-
* top-level task. The associated mechanics may be JVM-dependent
847-
* and must access particular Thread class fields to achieve this
848-
* effect.
834+
* InnocuousForkJoinWorkerThread for the commonPool. These
835+
* workers do not belong to any user-defined ThreadGroup, and
836+
* clear all ThreadLocals and reset the ContextClassLoader before
837+
* (re)activating to execute top-level tasks. The associated
838+
* mechanics may be JVM-dependent and must access particular
839+
* Thread class fields to achieve this effect.
849840
*
850841
* InterruptibleTasks
851842
* ====================
@@ -917,9 +908,6 @@ public class ForkJoinPool extends AbstractExecutorService {
917908
* shorts would suffice. For class WorkQueue, an
918909
* embedded @Contended region segregates fields most heavily
919910
* updated by owners from those most commonly read by stealers or
920-
* other management. For class WorkQueue, an embedded padded
921-
* region segregates fields (all declared as "int") most heavily
922-
* updated by owners from those most commonly read by stealers or
923911
* other management.
924912
*
925913
* Initial sizing and resizing of WorkQueue arrays is an even more
@@ -929,8 +917,10 @@ public class ForkJoinPool extends AbstractExecutorService {
929917
* direct false-sharing and indirect cases due to GC bookkeeping
930918
* (cardmarks etc), and reduce the number of resizes, which are
931919
* not especially fast because they require atomic transfers.
932-
* Currently, arrays are initialized to be just large enough to
933-
* avoid resizing in most tree-structured tasks. (Maintenance note:
920+
* Currently, arrays for workers are initialized to be just large
921+
* enough to avoid resizing in most tree-structured tasks, but
922+
* larger for external queues where both false-sharing problems
923+
* and the need for resizing are more common. (Maintenance note:
934924
* any changes in fields, queues, or their uses, or JVM layout
935925
* policies, must be accompanied by re-evaluation of these
936926
* placement and sizing decisions.)
@@ -1019,6 +1009,12 @@ public class ForkJoinPool extends AbstractExecutorService {
10191009
*/
10201010
static final int INITIAL_QUEUE_CAPACITY = 1 << 6;
10211011

1012+
/**
1013+
* Initial capacity of work-stealing queue array for external queues.
1014+
* Must be a power of two, at least 2. See above.
1015+
*/
1016+
static final int INITIAL_EXTERNAL_QUEUE_CAPACITY = 1 << 9;
1017+
10221018
// conversions among short, int, long
10231019
static final int SMASK = 0xffff; // (unsigned) short bits
10241020
static final long LMASK = 0xffffffffL; // lower 32 bits of long
@@ -1097,21 +1093,6 @@ static long slotOffset(int index) {
10971093
return ((long)index << ASHIFT) + ABASE;
10981094
}
10991095

1100-
/**
1101-
* If there is a security manager, makes sure caller has
1102-
* permission to modify threads.
1103-
*/
1104-
@SuppressWarnings("removal")
1105-
private static void checkPermission() {
1106-
SecurityManager security; RuntimePermission perm;
1107-
if ((security = System.getSecurityManager()) != null) {
1108-
if ((perm = modifyThreadPermission) == null)
1109-
modifyThreadPermission = perm = // races OK
1110-
new RuntimePermission("modifyThread");
1111-
security.checkPermission(perm);
1112-
}
1113-
}
1114-
11151096
// Nested classes
11161097

11171098
/**
@@ -1147,64 +1128,9 @@ public static interface ForkJoinWorkerThreadFactory {
11471128
static final class DefaultForkJoinWorkerThreadFactory
11481129
implements ForkJoinWorkerThreadFactory {
11491130
public final ForkJoinWorkerThread newThread(ForkJoinPool pool) {
1150-
boolean isCommon = (pool.workerNamePrefix == null);
1151-
@SuppressWarnings("removal")
1152-
SecurityManager sm = System.getSecurityManager();
1153-
if (sm != null && isCommon)
1154-
return newCommonWithACC(pool);
1155-
else
1156-
return newRegularWithACC(pool);
1157-
}
1158-
1159-
/*
1160-
* Create and use static AccessControlContexts only if there
1161-
* is a SecurityManager. (These can be removed if/when
1162-
* SecurityManagers are removed from platform.) The ACCs are
1163-
* immutable and equivalent even when racily initialized, so
1164-
* they don't require locking, although with the chance of
1165-
* needlessly duplicate construction.
1166-
*/
1167-
@SuppressWarnings("removal")
1168-
static volatile AccessControlContext regularACC, commonACC;
1169-
1170-
@SuppressWarnings("removal")
1171-
static ForkJoinWorkerThread newRegularWithACC(ForkJoinPool pool) {
1172-
AccessControlContext acc = regularACC;
1173-
if (acc == null) {
1174-
Permissions ps = new Permissions();
1175-
ps.add(new RuntimePermission("getClassLoader"));
1176-
ps.add(new RuntimePermission("setContextClassLoader"));
1177-
regularACC = acc =
1178-
new AccessControlContext(new ProtectionDomain[] {
1179-
new ProtectionDomain(null, ps) });
1180-
}
1181-
return AccessController.doPrivileged(
1182-
new PrivilegedAction<>() {
1183-
public ForkJoinWorkerThread run() {
1184-
return new ForkJoinWorkerThread(null, pool, true, false);
1185-
}}, acc);
1186-
}
1187-
1188-
@SuppressWarnings("removal")
1189-
static ForkJoinWorkerThread newCommonWithACC(ForkJoinPool pool) {
1190-
AccessControlContext acc = commonACC;
1191-
if (acc == null) {
1192-
Permissions ps = new Permissions();
1193-
ps.add(new RuntimePermission("getClassLoader"));
1194-
ps.add(new RuntimePermission("setContextClassLoader"));
1195-
ps.add(new RuntimePermission("modifyThread"));
1196-
ps.add(new RuntimePermission("enableContextClassLoaderOverride"));
1197-
ps.add(new RuntimePermission("modifyThreadGroup"));
1198-
commonACC = acc =
1199-
new AccessControlContext(new ProtectionDomain[] {
1200-
new ProtectionDomain(null, ps) });
1201-
}
1202-
return AccessController.doPrivileged(
1203-
new PrivilegedAction<>() {
1204-
public ForkJoinWorkerThread run() {
1205-
return new ForkJoinWorkerThread.
1206-
InnocuousForkJoinWorkerThread(pool);
1207-
}}, acc);
1131+
return ((pool.workerNamePrefix == null) ? // is commonPool
1132+
new ForkJoinWorkerThread.InnocuousForkJoinWorkerThread(pool) :
1133+
new ForkJoinWorkerThread(null, pool, true, false));
12081134
}
12091135
}
12101136

@@ -1264,7 +1190,9 @@ final boolean tryLockPhase() { // seqlock acquire
12641190
*/
12651191
WorkQueue(ForkJoinWorkerThread owner, int id, int cfg,
12661192
boolean clearThreadLocals) {
1267-
array = new ForkJoinTask<?>[INITIAL_QUEUE_CAPACITY];
1193+
array = new ForkJoinTask<?>[owner == null ?
1194+
INITIAL_EXTERNAL_QUEUE_CAPACITY :
1195+
INITIAL_QUEUE_CAPACITY];
12681196
this.owner = owner;
12691197
this.config = (clearThreadLocals) ? cfg | CLEAR_TLS : cfg;
12701198
}
@@ -3024,7 +2952,6 @@ public ForkJoinPool(int parallelism,
30242952
Predicate<? super ForkJoinPool> saturate,
30252953
long keepAliveTime,
30262954
TimeUnit unit) {
3027-
checkPermission();
30282955
int p = parallelism;
30292956
if (p <= 0 || p > MAX_CAP || p > maximumPoolSize || keepAliveTime <= 0L)
30302957
throw new IllegalArgumentException();
@@ -3312,7 +3239,6 @@ public int setParallelism(int size) {
33123239
throw new IllegalArgumentException();
33133240
if ((config & PRESET_SIZE) != 0)
33143241
throw new UnsupportedOperationException("Cannot override System property");
3315-
checkPermission();
33163242
return getAndSetParallelism(size);
33173243
}
33183244

@@ -3710,7 +3636,6 @@ public String toString() {
37103636
* may not be rejected.
37113637
*/
37123638
public void shutdown() {
3713-
checkPermission();
37143639
if (workerNamePrefix != null) // not common pool
37153640
tryTerminate(false, true);
37163641
}
@@ -3730,7 +3655,6 @@ public void shutdown() {
37303655
* @return an empty list
37313656
*/
37323657
public List<Runnable> shutdownNow() {
3733-
checkPermission();
37343658
if (workerNamePrefix != null) // not common pool
37353659
tryTerminate(true, true);
37363660
return Collections.emptyList();
@@ -3837,7 +3761,6 @@ public boolean awaitQuiescence(long timeout, TimeUnit unit) {
38373761
@Override
38383762
public void close() {
38393763
if (workerNamePrefix != null) {
3840-
checkPermission();
38413764
CountDownLatch done = null;
38423765
boolean interrupted = false;
38433766
while ((tryTerminate(interrupted, true) & TERMINATED) == 0) {
@@ -4075,11 +3998,6 @@ public void endCompensatedBlock(ForkJoinPool pool, long post) {
40753998
});
40763999
defaultForkJoinWorkerThreadFactory =
40774000
new DefaultForkJoinWorkerThreadFactory();
4078-
@SuppressWarnings("removal")
4079-
ForkJoinPool p = common = (System.getSecurityManager() == null) ?
4080-
new ForkJoinPool((byte)0) :
4081-
AccessController.doPrivileged(new PrivilegedAction<>() {
4082-
public ForkJoinPool run() {
4083-
return new ForkJoinPool((byte)0); }});
4001+
common = new ForkJoinPool((byte)0);
40844002
}
40854003
}

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

+25-25
Original file line numberDiff line numberDiff line change
@@ -35,10 +35,6 @@
3535

3636
package java.util.concurrent;
3737

38-
import java.security.AccessController;
39-
import java.security.AccessControlContext;
40-
import java.security.PrivilegedAction;
41-
import java.security.ProtectionDomain;
4238
import jdk.internal.access.JavaLangAccess;
4339
import jdk.internal.access.SharedSecrets;
4440
import jdk.internal.misc.Unsafe;
@@ -84,7 +80,7 @@ public class ForkJoinWorkerThread extends Thread {
8480
super.setDaemon(true);
8581
if (handler != null)
8682
super.setUncaughtExceptionHandler(handler);
87-
if (useSystemClassLoader)
83+
if (useSystemClassLoader && !clearThreadLocals) // else done by Thread ctor
8884
super.setContextClassLoader(ClassLoader.getSystemClassLoader());
8985
}
9086

@@ -228,18 +224,23 @@ static boolean hasKnownQueuedWork() {
228224
}
229225

230226
/**
231-
* Clears ThreadLocals, and if necessary resets ContextClassLoader
227+
* Clears ThreadLocals
232228
*/
233-
final void resetThreadLocals() {
229+
final void resetThreadLocals() {
234230
if (U.getReference(this, THREADLOCALS) != null)
235231
U.putReference(this, THREADLOCALS, null);
236232
if (U.getReference(this, INHERITABLETHREADLOCALS) != null)
237233
U.putReference(this, INHERITABLETHREADLOCALS, null);
238-
if ((this instanceof InnocuousForkJoinWorkerThread) &&
239-
((InnocuousForkJoinWorkerThread)this).needCCLReset())
240-
super.setContextClassLoader(ClassLoader.getSystemClassLoader());
234+
onThreadLocalReset();
241235
}
242236

237+
/**
238+
* Performs any further cleanup after ThreadLocals are cleared in
239+
* method resetThreadLocals
240+
*/
241+
void onThreadLocalReset() {
242+
}
243+
243244
private static final Unsafe U = Unsafe.getUnsafe();
244245
private static final long THREADLOCALS
245246
= U.objectFieldOffset(Thread.class, "threadLocals");
@@ -248,10 +249,10 @@ final void resetThreadLocals() {
248249
private static final JavaLangAccess JLA = SharedSecrets.getJavaLangAccess();
249250

250251
/**
251-
* A worker thread that has no permissions, is not a member of any
252-
* user-defined ThreadGroup, uses the system class loader as
253-
* thread context class loader, and clears all ThreadLocals after
254-
* running each top-level task.
252+
* A worker thread that is not a member of any user-defined
253+
* ThreadGroup, uses the system class loader as thread context
254+
* class loader, and clears all ThreadLocals after running each
255+
* top-level task.
255256
*/
256257
static final class InnocuousForkJoinWorkerThread extends ForkJoinWorkerThread {
257258
/** The ThreadGroup for all InnocuousForkJoinWorkerThreads */
@@ -264,21 +265,20 @@ static final class InnocuousForkJoinWorkerThread extends ForkJoinWorkerThread {
264265
@Override // to silently fail
265266
public void setUncaughtExceptionHandler(UncaughtExceptionHandler x) { }
266267

267-
@Override // paranoically
268-
@SuppressWarnings("removal")
268+
@Override // to record changes
269269
public void setContextClassLoader(ClassLoader cl) {
270-
if (System.getSecurityManager() != null &&
271-
cl != null && ClassLoader.getSystemClassLoader() != cl)
272-
throw new SecurityException("setContextClassLoader");
273-
resetCCL = true;
274-
super.setContextClassLoader(cl);
270+
if (ClassLoader.getSystemClassLoader() != cl) {
271+
resetCCL = true;
272+
super.setContextClassLoader(cl);
273+
}
275274
}
276275

277-
final boolean needCCLReset() { // get and clear
278-
boolean needReset;
279-
if (needReset = resetCCL)
276+
@Override // to re-establish CCL if necessary
277+
final void onThreadLocalReset() {
278+
if (resetCCL) {
280279
resetCCL = false;
281-
return needReset;
280+
super.setContextClassLoader(ClassLoader.getSystemClassLoader());
281+
}
282282
}
283283

284284
static ThreadGroup createGroup() {

1 commit comments

Comments
 (1)

openjdk-notifier[bot] commented on Nov 26, 2024

@openjdk-notifier[bot]
Please sign in to comment.