/* * Copyright (c) 2023, 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. */ /** * @test * @bug 8309034 * @summary Test that when saving a class initialization failure caused by * a StackOverflowError, that we record the SOE as the underlying * cause, even if we can't create the ExceptionInInitializerError * @requires os.simpleArch == "x64" * @comment The reproducer only fails in the desired way on x64. * @requires vm.flagless * @comment This test could easily be perturbed so don't allow flag settings. * * @run main/othervm -Xss160K -Xint TestStackOverflowDuringInit */ import java.io.ByteArrayOutputStream; import java.io.PrintStream; public class TestStackOverflowDuringInit { // The setup for this is somewhat intricate. We need to trigger a // StackOverflowError during execution of the static initializer // for a class, but we need there to be insufficient stack left // for the creation of the ExceptionInInitializerError that would // occur in that case. So we can't just recurse in a static initializer // as that would unwind all the way allowing plenty of stack for the // EIIE. Instead we recurse outside of a static initializer context // and have a finally clause that will trigger class initialization // of another class, which is where we will fail to create the EIIE. // Even then this is non-trivial, only the use of Long.valueOf from // the original reproducer seems to trigger SOE in just the right places. static void recurse() { try { // This will initialize Long but not touch LongCache. Long.valueOf(1024L); recurse(); } finally { // This will require initializing LongCache, which will // initially fail due to StackOverflowError and so LongCache // will be marked erroneous. As we unwind and again execute this // we will throw NoClassDefFoundError due to the erroneous // state of LongCache. Long.valueOf(0); } } public static void main(String[] args) throws Exception { String expected = "java.lang.NoClassDefFoundError: Could not initialize class java.lang.Long$LongCache"; String cause = "Caused by: java.lang.StackOverflowError"; try { recurse(); } catch (Throwable ex) { // ex.printStackTrace(); verify_stack(ex, expected, cause); } } private static void verify_stack(Throwable e, String expected, String cause) throws Exception { ByteArrayOutputStream byteOS = new ByteArrayOutputStream(); try (PrintStream printStream = new PrintStream(byteOS)) { e.printStackTrace(printStream); } String stackTrace = byteOS.toString("ASCII"); System.out.println(stackTrace); if (!stackTrace.contains(expected) || (cause != null && !stackTrace.contains(cause))) { throw new RuntimeException(expected + " and/or " + cause + " missing from stacktrace"); } } }