diff --git a/src/java.base/share/classes/sun/util/calendar/CalendarSystem.java b/src/java.base/share/classes/sun/util/calendar/CalendarSystem.java
index 71c7212c644..91a689f5f03 100644
--- a/src/java.base/share/classes/sun/util/calendar/CalendarSystem.java
+++ b/src/java.base/share/classes/sun/util/calendar/CalendarSystem.java
@@ -1,5 +1,5 @@
/*
- * Copyright (c) 2000, 2013, Oracle and/or its affiliates. All rights reserved.
+ * Copyright (c) 2000, 2021, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
@@ -111,7 +111,9 @@ public abstract class CalendarSystem {
}
}
- private static final Gregorian GREGORIAN_INSTANCE = new Gregorian();
+ private static final class GregorianHolder {
+ private static final Gregorian GREGORIAN_INSTANCE = new Gregorian();
+ }
/**
* Returns the singleton instance of the Gregorian
@@ -120,7 +122,7 @@ public abstract class CalendarSystem {
* @return the Gregorian instance
*/
public static Gregorian getGregorianCalendar() {
- return GREGORIAN_INSTANCE;
+ return GregorianHolder.GREGORIAN_INSTANCE;
}
/**
@@ -135,7 +137,7 @@ public abstract class CalendarSystem {
*/
public static CalendarSystem forName(String calendarName) {
if ("gregorian".equals(calendarName)) {
- return GREGORIAN_INSTANCE;
+ return GregorianHolder.GREGORIAN_INSTANCE;
}
if (!initialized) {
diff --git a/test/jdk/sun/util/calendar/CalendarSystemDeadLockTest.java b/test/jdk/sun/util/calendar/CalendarSystemDeadLockTest.java
new file mode 100644
index 00000000000..c40f78f7ade
--- /dev/null
+++ b/test/jdk/sun/util/calendar/CalendarSystemDeadLockTest.java
@@ -0,0 +1,177 @@
+/*
+ * Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
+ * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
+ *
+ * This code is free software; you can redistribute it and/or modify it
+ * under the terms of the GNU General Public License version 2 only, as
+ * published by the Free Software Foundation.
+ *
+ * This code is distributed in the hope that it will be useful, but WITHOUT
+ * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
+ * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
+ * version 2 for more details (a copy is included in the LICENSE file that
+ * accompanied this code).
+ *
+ * You should have received a copy of the GNU General Public License version
+ * 2 along with this work; if not, write to the Free Software Foundation,
+ * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
+ *
+ * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
+ * or visit www.oracle.com if you need additional information or have any
+ * questions.
+ */
+
+import java.util.ArrayList;
+import java.util.Collections;
+import java.util.List;
+import java.util.concurrent.Callable;
+import java.util.concurrent.CountDownLatch;
+import java.util.concurrent.ExecutorService;
+import java.util.concurrent.Executors;
+import java.util.concurrent.Future;
+
+/**
+ * @test
+ * @bug 8273790
+ * @summary Verify that concurrent classloading of sun.util.calendar.Gregorian and
+ * sun.util.calendar.CalendarSystem doesn't lead to a deadlock
+ * @modules java.base/sun.util.calendar:open
+ * @run main/othervm CalendarSystemDeadLockTest
+ * @run main/othervm CalendarSystemDeadLockTest
+ * @run main/othervm CalendarSystemDeadLockTest
+ * @run main/othervm CalendarSystemDeadLockTest
+ * @run main/othervm CalendarSystemDeadLockTest
+ */
+public class CalendarSystemDeadLockTest {
+
+ public static void main(final String[] args) throws Exception {
+ testConcurrentClassLoad();
+ }
+
+ /**
+ * Loads {@code sun.util.calendar.Gregorian} and {@code sun.util.calendar.CalendarSystem}
+ * and invokes {@code sun.util.calendar.CalendarSystem#getGregorianCalendar()} concurrently
+ * in a thread of their own and expects the classloading of both those classes
+ * to succeed. Additionally, after these tasks are done, calls the
+ * sun.util.calendar.CalendarSystem#getGregorianCalendar() and expects it to return a singleton
+ * instance
+ */
+ private static void testConcurrentClassLoad() throws Exception {
+ final int numTasks = 7;
+ final CountDownLatch taskTriggerLatch = new CountDownLatch(numTasks);
+ final List> tasks = new ArrayList<>();
+ // add the sun.util.calendar.Gregorian and sun.util.calendar.CalendarSystem for classloading.
+ // there are main 2 classes which had a cyclic call in their static init
+ tasks.add(new ClassLoadTask("sun.util.calendar.Gregorian", taskTriggerLatch));
+ tasks.add(new ClassLoadTask("sun.util.calendar.CalendarSystem", taskTriggerLatch));
+ // add a few other classes for classloading, those which call CalendarSystem#getGregorianCalendar()
+ // or CalendarSystem#forName() during their static init
+ tasks.add(new ClassLoadTask("java.util.GregorianCalendar", taskTriggerLatch));
+ tasks.add(new ClassLoadTask("java.util.Date", taskTriggerLatch));
+ tasks.add(new ClassLoadTask("java.util.JapaneseImperialCalendar", taskTriggerLatch));
+ // add a couple of tasks which directly invoke sun.util.calendar.CalendarSystem#getGregorianCalendar()
+ tasks.add(new GetGregorianCalTask(taskTriggerLatch));
+ tasks.add(new GetGregorianCalTask(taskTriggerLatch));
+ // before triggering the tests make sure we have created the correct number of tasks
+ // the countdown latch uses/expects
+ if (numTasks != tasks.size()) {
+ throw new RuntimeException("Test setup failure - unexpected number of tasks " + tasks.size()
+ + ", expected " + numTasks);
+ }
+ final ExecutorService executor = Executors.newFixedThreadPool(tasks.size());
+ try {
+ final Future>[] results = new Future[tasks.size()];
+ // submit
+ int i = 0;
+ for (final Callable> task : tasks) {
+ results[i++] = executor.submit(task);
+ }
+ // wait for completion
+ for (i = 0; i < tasks.size(); i++) {
+ results[i].get();
+ }
+ } finally {
+ executor.shutdownNow();
+ }
+ // check that the sun.util.calendar.CalendarSystem#getGregorianCalendar() does indeed return
+ // a proper instance
+ final Object gCal = callCalSystemGetGregorianCal();
+ if (gCal == null) {
+ throw new RuntimeException("sun.util.calendar.CalendarSystem#getGregorianCalendar()" +
+ " unexpectedly returned null");
+ }
+ // now verify that each call to getGregorianCalendar(), either in the tasks or here, returned the exact
+ // same instance
+ if (GetGregorianCalTask.instances.size() != 2) {
+ throw new RuntimeException("Unexpected number of results from call " +
+ "to sun.util.calendar.CalendarSystem#getGregorianCalendar()");
+ }
+ // intentional identity check since sun.util.calendar.CalendarSystem#getGregorianCalendar() is
+ // expected to return a singleton instance
+ if ((gCal != GetGregorianCalTask.instances.get(0)) || (gCal != GetGregorianCalTask.instances.get(1))) {
+ throw new RuntimeException("sun.util.calendar.CalendarSystem#getGregorianCalendar()" +
+ " returned different instances");
+ }
+ }
+
+ /**
+ * Reflectively calls sun.util.calendar.CalendarSystem#getGregorianCalendar() and returns
+ * the result
+ */
+ private static Object callCalSystemGetGregorianCal() throws Exception {
+ final Class> k = Class.forName("sun.util.calendar.CalendarSystem");
+ return k.getDeclaredMethod("getGregorianCalendar").invoke(null);
+ }
+
+ private static class ClassLoadTask implements Callable> {
+ private final String className;
+ private final CountDownLatch latch;
+
+ private ClassLoadTask(final String className, final CountDownLatch latch) {
+ this.className = className;
+ this.latch = latch;
+ }
+
+ @Override
+ public Class> call() {
+ System.out.println(Thread.currentThread().getName() + " loading " + this.className);
+ try {
+ // let the other tasks know we are ready to trigger our work
+ latch.countDown();
+ // wait for the other task to let us know they are ready to trigger their work too
+ latch.await();
+ return Class.forName(this.className);
+ } catch (Exception e) {
+ throw new RuntimeException(e);
+ }
+ }
+ }
+
+ private static class GetGregorianCalTask implements Callable