/*
* 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
*
* @library /test/lib
* @requires vm.hasSA
* @modules jdk.hotspot.agent/sun.jvm.hotspot
* jdk.hotspot.agent/sun.jvm.hotspot.debugger
* jdk.hotspot.agent/sun.jvm.hotspot.types
* jdk.hotspot.agent/sun.jvm.hotspot.types.basic
*
* @run driver UniqueVtableTest
*/
import java.util.ArrayList;
import java.util.Iterator;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import sun.jvm.hotspot.HotSpotAgent;
import sun.jvm.hotspot.debugger.Address;
import sun.jvm.hotspot.types.Type;
import sun.jvm.hotspot.types.basic.BasicTypeDataBase;
import jdk.test.lib.apps.LingeredApp;
import jdk.test.lib.process.ProcessTools;
import jdk.test.lib.process.OutputAnalyzer;
import jdk.test.lib.SA.SATestUtils;
public class UniqueVtableTest {
private static String type2String(Type t) {
return t + " (extends " + t.getSuperclass() + ")";
}
private static void log(Object o) {
System.out.println(o);
}
private static void runTest(long pid) throws Throwable {
HotSpotAgent agent = new HotSpotAgent();
log("Attaching to process ID " + pid + "...");
agent.attach((int) pid);
log("Attached successfully.");
Throwable reasonToFail = null;
try {
runTest(agent);
} catch (Throwable ex) {
reasonToFail = ex;
} finally {
try {
agent.detach();
} catch (Exception ex) {
log("detach error:");
ex.printStackTrace(System.out);
// do not override original error
if (reasonToFail != null) {
reasonToFail = ex;
}
}
}
if (reasonToFail != null) {
throw reasonToFail;
}
}
private static void runTest(HotSpotAgent agent) throws Throwable {
Map
> vtableToTypesMap = new HashMap<>();
Iterator it = agent.getTypeDataBase().getTypes();
int dupsFound = 0;
// TypeDataBase knows nothing about vtables,
// but actually agent.getTypeDataBase() returns HotSpotTypeDataBase (extends BasicTypeDataBase)
// and BasicTypeDataBase has a method to get vtable for Types.
BasicTypeDataBase typeDB = (BasicTypeDataBase)(agent.getTypeDataBase());
int total = 0;
int vm_classes_with_vtable = 0;
int vm_classes_without_vtable = 0;
while (it.hasNext()) {
total++;
Type t = it.next();
Address vtable = typeDB.vtblForType(t);
if (vtable != null) {
vm_classes_with_vtable++;
List typeList = vtableToTypesMap.get(vtable);
if (typeList == null) {
vtableToTypesMap.put(vtable, new ArrayList<>(List.of(t)));
} else {
// duplicate found
dupsFound++;
typeList.add(t);
}
}
// IntegerType/StringType/JavaPrimitiveType/OopType/PointerType types
// are expected to have no vtable.
// Log classes which might need vtable.
if (vtable == null
&& !t.isCIntegerType()
&& !t.isCStringType()
&& !t.isJavaPrimitiveType()
&& !t.isOopType()
&& !t.isPointerType()) {
vm_classes_without_vtable++;
log("vtable is null for " + type2String(t));
}
}
log("total: " + total
+ ", vm_classes_with_vtable: " + vm_classes_with_vtable
+ ", vm_classes_without_vtable: " + vm_classes_without_vtable);
if (dupsFound > 0) {
vtableToTypesMap.forEach((vtable, list) -> {
if (list.size() > 1) {
log("Duplicate vtable: " + vtable + ": ");
list.forEach(t -> log(" - " + type2String(t)));
}
});
throw new RuntimeException("Duplicate vtable(s) found: " + dupsFound);
}
}
private static void createAnotherToAttach(long lingeredAppPid) throws Throwable {
// Start a new process to attach to the lingered app
ProcessBuilder processBuilder = ProcessTools.createJavaProcessBuilder(
"--add-modules=jdk.hotspot.agent",
"--add-exports=jdk.hotspot.agent/sun.jvm.hotspot=ALL-UNNAMED",
"--add-exports=jdk.hotspot.agent/sun.jvm.hotspot.debugger=ALL-UNNAMED",
"--add-exports=jdk.hotspot.agent/sun.jvm.hotspot.types=ALL-UNNAMED",
"--add-exports=jdk.hotspot.agent/sun.jvm.hotspot.types.basic=ALL-UNNAMED",
"UniqueVtableTest",
Long.toString(lingeredAppPid));
SATestUtils.addPrivilegesIfNeeded(processBuilder);
OutputAnalyzer output = ProcessTools.executeProcess(processBuilder);
output.shouldHaveExitValue(0);
System.out.println(output.getOutput());
}
private static void runMain() throws Throwable {
Throwable reasonToFail = null;
LingeredApp app = null;
try {
app = LingeredApp.startApp();
createAnotherToAttach(app.getPid());
} catch (Throwable ex) {
reasonToFail = ex;
} finally {
try {
LingeredApp.stopApp(app);
} catch (Exception ex) {
log("LingeredApp.stopApp error:");
ex.printStackTrace(System.out);
// do not override original error
if (reasonToFail != null) {
reasonToFail = ex;
}
}
}
if (reasonToFail != null) {
throw reasonToFail;
}
}
public static void main(String... args) throws Throwable {
SATestUtils.skipIfCannotAttach(); // throws SkippedException if attach not expected to work.
if (args == null || args.length == 0) {
// Main test process.
runMain();
} else {
// Sub-process to attach, arg[0] is the target process pid.
runTest(Long.parseLong(args[0]));
}
}
}