2023-03-09 19:13:02 +00:00
|
|
|
/*
|
|
|
|
* 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
|
|
|
|
*
|
2023-03-17 18:23:03 +00:00
|
|
|
* @run driver UniqueVtableTest
|
2023-03-09 19:13:02 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
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;
|
2023-03-17 18:23:03 +00:00
|
|
|
import jdk.test.lib.process.ProcessTools;
|
|
|
|
import jdk.test.lib.process.OutputAnalyzer;
|
2023-03-09 19:13:02 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2023-03-17 18:23:03 +00:00
|
|
|
private static void runTest(long pid) throws Throwable {
|
|
|
|
HotSpotAgent agent = new HotSpotAgent();
|
2023-03-09 19:13:02 +00:00
|
|
|
log("Attaching to process ID " + pid + "...");
|
|
|
|
agent.attach((int) pid);
|
|
|
|
log("Attached successfully.");
|
|
|
|
|
2023-03-17 18:23:03 +00:00
|
|
|
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;
|
2023-03-09 19:13:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-17 18:23:03 +00:00
|
|
|
private static void runTest(HotSpotAgent agent) throws Throwable {
|
2023-03-09 19:13:02 +00:00
|
|
|
Map<Address, List<Type>> vtableToTypesMap = new HashMap<>();
|
|
|
|
Iterator<Type> 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<Type> 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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-17 18:23:03 +00:00
|
|
|
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 {
|
2023-03-09 19:13:02 +00:00
|
|
|
Throwable reasonToFail = null;
|
|
|
|
LingeredApp app = null;
|
|
|
|
try {
|
|
|
|
app = LingeredApp.startApp();
|
2023-03-17 18:23:03 +00:00
|
|
|
createAnotherToAttach(app.getPid());
|
2023-03-09 19:13:02 +00:00
|
|
|
} 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.
|
|
|
|
|
2023-03-17 18:23:03 +00:00
|
|
|
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]));
|
|
|
|
}
|
2023-03-09 19:13:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|