Skip to content

Commit 5e59d28

Browse files
committedApr 1, 2021
8262046: Clean up parallel class loading code and comments
Reviewed-by: lfoltan, iklam
1 parent 04f24fe commit 5e59d28

File tree

11 files changed

+722
-229
lines changed

11 files changed

+722
-229
lines changed
 

‎src/hotspot/share/classfile/systemDictionary.cpp

+182-212
Large diffs are not rendered by default.

‎src/hotspot/share/classfile/systemDictionary.hpp

+14-17
Original file line numberDiff line numberDiff line change
@@ -46,9 +46,7 @@
4646
// be done concurrently, but only by different loaders.
4747
//
4848
// During loading a placeholder (name, loader) is temporarily placed in
49-
// a side data structure, and is used to detect ClassCircularityErrors
50-
// and to perform verification during GC. A GC can occur in the midst
51-
// of class loading, as we call out to Java, have to take locks, etc.
49+
// a side data structure, and is used to detect ClassCircularityErrors.
5250
//
5351
// When class loading is finished, a new entry is added to the dictionary
5452
// of the class loader and the placeholder is removed. Note that the protection
@@ -58,9 +56,8 @@
5856
// Clients of this class who are interested in finding if a class has
5957
// been completely loaded -- not classes in the process of being loaded --
6058
// can read the dictionary unlocked. This is safe because
61-
// - entries are only deleted at safepoints
62-
// - readers cannot come to a safepoint while actively examining
63-
// an entry (an entry cannot be deleted from under a reader)
59+
// - entries are only deleted when the class loader is not alive, when the
60+
// entire dictionary is deleted.
6461
// - entries must be fully formed before they are available to concurrent
6562
// readers (we must ensure write ordering)
6663
//
@@ -340,21 +337,21 @@ class SystemDictionary : AllStatic {
340337
static Klass* resolve_array_class_or_null(Symbol* class_name,
341338
Handle class_loader,
342339
Handle protection_domain, TRAPS);
343-
static InstanceKlass* handle_parallel_super_load(Symbol* class_name,
344-
Symbol* supername,
345-
Handle class_loader,
346-
Handle protection_domain,
347-
Handle lockObject, TRAPS);
348-
// Wait on SystemDictionary_lock; unlocks lockObject before
349-
// waiting; relocks lockObject with correct recursion count
350-
// after waiting, but before reentering SystemDictionary_lock
351-
// to preserve lock order semantics.
352-
static void double_lock_wait(JavaThread* thread, Handle lockObject);
340+
static InstanceKlass* handle_parallel_loading(JavaThread* current,
341+
unsigned int name_hash,
342+
Symbol* name,
343+
ClassLoaderData* loader_data,
344+
Handle lockObject,
345+
bool* throw_circularity_error);
346+
353347
static void define_instance_class(InstanceKlass* k, Handle class_loader, TRAPS);
354348
static InstanceKlass* find_or_define_helper(Symbol* class_name,
355349
Handle class_loader,
356350
InstanceKlass* k, TRAPS);
357-
static InstanceKlass* load_instance_class(Symbol* class_name, Handle class_loader, TRAPS);
351+
static InstanceKlass* load_instance_class_impl(Symbol* class_name, Handle class_loader, TRAPS);
352+
static InstanceKlass* load_instance_class(unsigned int name_hash,
353+
Symbol* class_name,
354+
Handle class_loader, TRAPS);
358355

359356
static bool is_shared_class_visible(Symbol* class_name, InstanceKlass* ik,
360357
PackageEntry* pkg_entry,
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
import jdk.internal.org.objectweb.asm.*;
25+
26+
public class AsmClasses implements Opcodes {
27+
28+
// class B extends A {} to generate ClassCircularityError
29+
// Can't use jcod, jasm or InMemoryCompiler because the compiler will
30+
// see that A extends B and fail.
31+
public static byte[] dumpB() throws Exception {
32+
33+
ClassWriter classWriter = new ClassWriter(0);
34+
FieldVisitor fieldVisitor;
35+
RecordComponentVisitor recordComponentVisitor;
36+
MethodVisitor methodVisitor;
37+
AnnotationVisitor annotationVisitor0;
38+
39+
classWriter.visit(61, ACC_PUBLIC | ACC_SUPER, "B", null, "A", null);
40+
41+
classWriter.visitSource("B.java", null);
42+
methodVisitor = classWriter.visitMethod(ACC_PUBLIC, "<init>", "()V", null, null);
43+
methodVisitor.visitCode();
44+
Label label0 = new Label();
45+
methodVisitor.visitLabel(label0);
46+
methodVisitor.visitLineNumber(24, label0);
47+
methodVisitor.visitVarInsn(ALOAD, 0);
48+
methodVisitor.visitMethodInsn(INVOKESPECIAL, "A", "<init>", "()V", false);
49+
methodVisitor.visitInsn(RETURN);
50+
methodVisitor.visitMaxs(1, 1);
51+
methodVisitor.visitEnd();
52+
classWriter.visitEnd();
53+
54+
return classWriter.toByteArray();
55+
}
56+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
class ClassLoadingThread extends Thread {
25+
26+
private ClassLoader ldr = null;
27+
private Object thread_sync = null;
28+
29+
public ClassLoadingThread(ClassLoader loader, Object sync) {
30+
ldr = loader;
31+
thread_sync = sync;
32+
}
33+
34+
private boolean success = true;
35+
public boolean report_success() { return success; }
36+
37+
public void run() {
38+
try {
39+
ThreadPrint.println("Starting...");
40+
// Initiate class loading using specified type
41+
Class<?> a = Class.forName("ClassInLoader", true, ldr);
42+
Object obj = a.getConstructor().newInstance();
43+
44+
} catch (Throwable e) {
45+
ThreadPrint.println("Exception is caught: " + e);
46+
e.printStackTrace();
47+
success = false;
48+
} finally {
49+
ThreadPrint.println("Finished");
50+
// Wake up the second thread
51+
synchronized (thread_sync) {
52+
thread_sync.notify();
53+
}
54+
}
55+
}
56+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,151 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
import java.io.*;
25+
import jdk.test.lib.classloader.ClassUnloadCommon;
26+
27+
// This class loader will deadlock where one thread has a lock for A, trying to get a lock for B
28+
// and the other thread has a lock for B, trying to get a lock for A in the case of
29+
// A extends B extends A. It should throw ClassCircularityError from both threads.
30+
31+
class MyLoader extends ClassLoader {
32+
static {
33+
registerAsParallelCapable();
34+
}
35+
36+
private static boolean first = true;
37+
38+
public Class loadClass(String name) throws ClassNotFoundException {
39+
// Wait before getting B lock.
40+
if (name.equals("B") && first) {
41+
first = false;
42+
makeThreadWait();
43+
}
44+
synchronized(getClassLoadingLock(name)) {
45+
Class<?> c = findLoadedClass(name);
46+
if (c != null) return c;
47+
48+
if (name.equals("B") && !first) {
49+
wakeUpThread();
50+
}
51+
52+
byte[] b = loadClassData(name);
53+
if (b != null) {
54+
return defineClass(name, b, 0, b.length);
55+
} else {
56+
return super.loadClass(name);
57+
}
58+
}
59+
}
60+
61+
private static boolean parallel = false;
62+
private Object sync = new Object();
63+
private Object thread_sync = new Object();
64+
65+
private void makeThreadWait() {
66+
if (!parallel) { return; }
67+
68+
// Wake up the second thread here.
69+
synchronized (thread_sync) {
70+
thread_sync.notify();
71+
}
72+
if (isRegisteredAsParallelCapable()) {
73+
synchronized(sync) {
74+
try {
75+
ThreadPrint.println("t1 waits parallelCapable loader");
76+
sync.wait(); // Give up lock before request to load B
77+
} catch (InterruptedException e) {}
78+
}
79+
} else {
80+
try {
81+
ThreadPrint.println("t1 waits non-parallelCapable loader");
82+
wait(); // Give up lock before request to load B
83+
} catch (InterruptedException e) {}
84+
}
85+
}
86+
87+
// Parallel capable loader should wake up the first thread.
88+
// Non-parallelCapable class loader thread will be woken up by the jvm.
89+
private void wakeUpThread() {
90+
if (isRegisteredAsParallelCapable()) {
91+
synchronized(sync) {
92+
sync.notify();
93+
}
94+
}
95+
}
96+
97+
private byte[] loadClassData(String name) {
98+
if (name.equals("A")) {
99+
ThreadPrint.println("loading A extends B");
100+
return ClassUnloadCommon.getClassData("A");
101+
} else if (name.equals("B")) {
102+
ThreadPrint.println("loading B extends A");
103+
try {
104+
return AsmClasses.dumpB();
105+
} catch (Exception e) {
106+
e.printStackTrace();
107+
return null;
108+
}
109+
} else if (!name.startsWith("java")) {
110+
return ClassUnloadCommon.getClassData(name);
111+
}
112+
return null;
113+
}
114+
115+
116+
ClassLoadingThread[] threads = new ClassLoadingThread[2];
117+
private boolean success = true;
118+
119+
public boolean report_success() {
120+
for (int i = 0; i < 2; i++) {
121+
try {
122+
threads[i].join();
123+
if (!threads[i].report_success()) success = false;
124+
} catch (InterruptedException e) {}
125+
}
126+
return success;
127+
}
128+
129+
void startLoading() {
130+
131+
for (int i = 0; i < 2; i++) {
132+
threads[i] = new ClassLoadingThread(this, thread_sync);
133+
threads[i].setName("Loading Thread #" + (i + 1));
134+
threads[i].start();
135+
System.out.println("Thread " + (i + 1) + " was started...");
136+
// wait to start the second thread if not concurrent
137+
if (i == 0) {
138+
synchronized(thread_sync) {
139+
try {
140+
ThreadPrint.println("t2 waits");
141+
thread_sync.wait();
142+
} catch (InterruptedException e) {}
143+
}
144+
}
145+
}
146+
}
147+
148+
MyLoader(boolean load_in_parallel) {
149+
parallel = load_in_parallel;
150+
}
151+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
class MyNonParallelLoader extends MyLoader {
25+
// This loader isn't parallel capable because it's not registered in the static
26+
// initializer as such. parallelCapable is not an inheritable attribute.
27+
MyNonParallelLoader(boolean load_in_parallel) {
28+
super(load_in_parallel);
29+
}
30+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
/*
25+
* @test MyLoaderTest
26+
* @bug 8262046
27+
* @summary Call handle_parallel_super_load, loading parallel threads that throw CCE
28+
* @modules java.base/jdk.internal.org.objectweb.asm
29+
* java.base/jdk.internal.misc
30+
* @library /test/lib
31+
* @compile -XDignore.symbol.file AsmClasses.java
32+
* @compile test-classes/ClassInLoader.java test-classes/A.java test-classes/B.java
33+
* @run main/othervm ParallelSuperTest
34+
* @run main/othervm ParallelSuperTest -parallel
35+
* @run main/othervm ParallelSuperTest -parallel -parallelCapable
36+
*/
37+
38+
public class ParallelSuperTest {
39+
public static void main(java.lang.String[] args) throws Exception {
40+
boolean parallel = false;
41+
boolean parallelCapable = false;
42+
boolean success = true;
43+
for (int i = 0; i < args.length; i++) {
44+
try {
45+
// Don't print debug info
46+
if (args[i].equals("-parallel")) {
47+
parallel = true;
48+
} else if (args[i].equals("-parallelCapable")) {
49+
parallelCapable = true;
50+
} else {
51+
System.out.println("Unrecognized " + args[i]);
52+
}
53+
} catch (NumberFormatException e) {
54+
System.err.println("Invalid parameter: " + args[i - 1] + " " + args[i]);
55+
}
56+
}
57+
// The -parallel -parallelCapable case will deadlock on locks for A and B if
58+
// the jvm doesn't eagerly try to load A's superclass from the second thread.
59+
// ie. needs to call SystemDictionary::handle_parallel_super_load
60+
if (parallelCapable) {
61+
MyLoader ldr = new MyLoader(parallel);
62+
ldr.startLoading();
63+
success = ldr.report_success();
64+
} else {
65+
MyNonParallelLoader ldr = new MyNonParallelLoader(parallel);
66+
ldr.startLoading();
67+
success = ldr.report_success();
68+
}
69+
if (success) {
70+
System.out.println("PASSED");
71+
} else {
72+
throw new RuntimeException("FAILED");
73+
}
74+
}
75+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,31 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
25+
// class to print thread-annotated output
26+
class ThreadPrint {
27+
public static void println(String s) {
28+
System.out.println(Thread.currentThread().getName() + ": " + s);
29+
System.out.flush();
30+
}
31+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
public class A extends B {
25+
public A() {
26+
System.out.println("A extends B");
27+
throw new RuntimeException("Should throw CCE here");
28+
}
29+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
public class B { public B() { System.out.println("B"); } }
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
// Create a class to load inside the loader instance, that will load
25+
// A through a constant pool reference.
26+
// Class A extends B extends A
27+
class CP1 {
28+
void foo() throws Exception {
29+
ThreadPrint.println("CP1.foo()");
30+
try {
31+
Class<?> a = A.class;
32+
Object obj = a.getConstructor().newInstance();
33+
throw new RuntimeException("Should throw CCE here");
34+
} catch (Throwable e) {
35+
ThreadPrint.println("Exception is caught: " + e);
36+
if (!(e instanceof java.lang.ClassCircularityError)) {
37+
throw new RuntimeException("Unexpected exception");
38+
}
39+
}
40+
}
41+
}
42+
43+
// This class has a constant pool reference to B which will also get CCE, but
44+
// starting at B.
45+
class CP2 {
46+
void foo() throws Exception {
47+
ThreadPrint.println("CP2.foo()");
48+
try {
49+
Class<?> a = B.class;
50+
Object obj = a.getConstructor().newInstance();
51+
throw new RuntimeException("Should throw CCE here");
52+
} catch (Throwable e) {
53+
ThreadPrint.println("Exception is caught: " + e);
54+
if (!(e instanceof java.lang.ClassCircularityError)) {
55+
throw new RuntimeException("Unexpected exception");
56+
}
57+
}
58+
}
59+
}
60+
61+
public class ClassInLoader {
62+
private static boolean first = true;
63+
public ClassInLoader() throws Exception {
64+
ThreadPrint.println("ClassInLoader");
65+
if (first) {
66+
first = false;
67+
CP1 c1 = new CP1();
68+
c1.foo();
69+
} else {
70+
CP2 c2 = new CP2();
71+
c2.foo();
72+
}
73+
}
74+
}

0 commit comments

Comments
 (0)
Please sign in to comment.