/* * Copyright (c) 2020, 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.lang.instrument.Instrumentation; public class LockDuringDumpAgent implements Runnable { static boolean threadStarted = false; static Object lock = new Object(); // The following literal string will be stored into the VM's interned string table when this // class (or the LockDuringDumpApp class) is loaded during -Xshare:dump. As a result it will be // stored in the CDS archived heap (all strings in the dump-time interned string table are archived). // // We try to make sure this string is locked while the archived heap is dumped. CDS should // clear the lock states in this string's object header. See JDK-8249276. // // At run time, when LockDuringDumpApp loads this literal string (from the shared string table) // it should be able to lock it without problems. static String LITERAL = "@@LockDuringDump@@LITERAL"; // must be the same as in LockDuringDumpAgent public static void premain(String agentArg, Instrumentation instrumentation) { System.out.println("inside LockDuringDumpAgent: " + LockDuringDumpAgent.class.getClassLoader()); Thread t = new Thread(new LockDuringDumpAgent()); t.setDaemon(true); t.start(); waitForThreadStart(); } static void waitForThreadStart() { try { synchronized (lock) { while (!threadStarted) { lock.wait(); } System.out.println("Thread has started"); } } catch (Throwable t) { System.err.println("Unexpected: " + t); throw new RuntimeException(t); } } public void run() { try { synchronized (LITERAL) { System.out.println("Let's hold the lock on the literal string \"" + LITERAL + "\" + forever ....."); synchronized (lock) { threadStarted = true; lock.notifyAll(); } //if (false) { while (true) { Thread.sleep(1); } //} } } catch (Throwable t) { System.err.println("Unexpected: " + t); throw new RuntimeException(t); } } }