2016-02-18 23:26:43 +09:00
|
|
|
/*
|
2021-03-13 14:51:53 +00:00
|
|
|
* Copyright (c) 2016, 2021, Oracle and/or its affiliates. All rights reserved.
|
2016-02-18 23:26:43 +09:00
|
|
|
* 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.io.*;
|
|
|
|
import java.nio.file.*;
|
2018-02-01 21:03:37 -05:00
|
|
|
import java.util.Arrays;
|
2016-03-02 17:08:26 +03:00
|
|
|
import java.util.jar.Attributes;
|
|
|
|
import java.util.jar.JarEntry;
|
|
|
|
import java.util.jar.JarOutputStream;
|
|
|
|
import java.util.jar.Manifest;
|
2018-02-01 21:03:37 -05:00
|
|
|
import java.util.List;
|
|
|
|
import java.util.regex.Matcher;
|
|
|
|
import java.util.regex.Pattern;
|
2016-08-19 10:06:30 -04:00
|
|
|
import jdk.test.lib.Platform;
|
|
|
|
import jdk.test.lib.process.OutputAnalyzer;
|
2016-02-18 23:26:43 +09:00
|
|
|
import jdk.test.lib.dcmd.*;
|
|
|
|
import org.testng.annotations.Test;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Test to attach JVMTI java agent.
|
|
|
|
*
|
|
|
|
* @test
|
|
|
|
* @bug 8147388
|
2016-08-19 10:06:30 -04:00
|
|
|
* @library /test/lib
|
2016-04-09 23:03:39 +01:00
|
|
|
* @modules java.base/jdk.internal.misc
|
2016-02-18 23:26:43 +09:00
|
|
|
* java.compiler
|
|
|
|
* java.instrument
|
|
|
|
* java.management
|
2017-01-19 10:56:32 -05:00
|
|
|
* jdk.internal.jvmstat/sun.jvmstat.monitor
|
2016-08-19 10:06:30 -04:00
|
|
|
* @build SimpleJvmtiAgent
|
2021-03-13 14:51:53 +00:00
|
|
|
* @run driver jdk.test.lib.helpers.ClassFileInstaller SimpleJvmtiAgent
|
2017-11-16 14:47:31 -08:00
|
|
|
* @run testng/othervm LoadAgentDcmdTest
|
2016-02-18 23:26:43 +09:00
|
|
|
*/
|
|
|
|
public class LoadAgentDcmdTest {
|
|
|
|
|
|
|
|
public String getLibInstrumentPath() throws FileNotFoundException {
|
|
|
|
String jdkPath = System.getProperty("test.jdk");
|
|
|
|
|
|
|
|
if (jdkPath == null) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"System property 'test.jdk' not set. " +
|
|
|
|
"This property is normally set by jtreg. " +
|
|
|
|
"When running test separately, set this property using " +
|
|
|
|
"'-Dtest.jdk=/path/to/jdk'.");
|
|
|
|
}
|
|
|
|
|
2016-08-19 10:06:30 -04:00
|
|
|
Path libpath = Paths.get(jdkPath, jdkLibPath(), sharedObjectName("instrument"));
|
2016-02-18 23:26:43 +09:00
|
|
|
|
|
|
|
if (!libpath.toFile().exists()) {
|
|
|
|
throw new FileNotFoundException(
|
|
|
|
"Could not find " + libpath.toAbsolutePath());
|
|
|
|
}
|
|
|
|
|
|
|
|
return libpath.toAbsolutePath().toString();
|
|
|
|
}
|
|
|
|
|
2016-03-02 17:08:26 +03:00
|
|
|
|
|
|
|
public void createJarFileForAgent()
|
|
|
|
throws IOException {
|
|
|
|
|
|
|
|
final String jarName = "agent.jar";
|
|
|
|
final String agentClass = "SimpleJvmtiAgent";
|
|
|
|
|
|
|
|
Manifest manifest = new Manifest();
|
|
|
|
|
|
|
|
manifest.getMainAttributes().put(
|
|
|
|
Attributes.Name.MANIFEST_VERSION, "1.0");
|
|
|
|
|
|
|
|
manifest.getMainAttributes().put(
|
|
|
|
new Attributes.Name("Agent-Class"), agentClass);
|
|
|
|
|
|
|
|
JarOutputStream target = null;
|
|
|
|
|
|
|
|
try {
|
|
|
|
target = new
|
|
|
|
JarOutputStream(new FileOutputStream(jarName), manifest);
|
|
|
|
JarEntry entry = new JarEntry(agentClass + ".class");
|
|
|
|
target.putNextEntry(entry);
|
|
|
|
target.closeEntry();
|
|
|
|
} finally {
|
|
|
|
target.close();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-01 21:03:37 -05:00
|
|
|
static void checkWarningsOnly(OutputAnalyzer out) {
|
|
|
|
// stderr should be empty except for VM warnings.
|
|
|
|
if (!out.getStderr().isEmpty()) {
|
|
|
|
List<String> lines = Arrays.asList(out.getStderr().split("(\\r\\n|\\n|\\r)"));
|
|
|
|
Pattern p = Pattern.compile(".*VM warning.*");
|
|
|
|
for (String line : lines) {
|
|
|
|
Matcher m = p.matcher(line);
|
|
|
|
if (!m.matches()) {
|
|
|
|
throw new RuntimeException("Stderr has output other than VM warnings");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-18 23:26:43 +09:00
|
|
|
public void run(CommandExecutor executor) {
|
|
|
|
try{
|
2016-03-02 17:08:26 +03:00
|
|
|
|
|
|
|
createJarFileForAgent();
|
2016-02-18 23:26:43 +09:00
|
|
|
|
|
|
|
String libpath = getLibInstrumentPath();
|
2016-03-02 17:08:26 +03:00
|
|
|
OutputAnalyzer output = null;
|
|
|
|
|
|
|
|
// Test 1: Native agent, no arguments
|
|
|
|
output = executor.execute("JVMTI.agent_load " +
|
|
|
|
libpath + " agent.jar");
|
2018-02-01 21:03:37 -05:00
|
|
|
checkWarningsOnly(output);
|
2016-02-18 23:26:43 +09:00
|
|
|
|
2016-03-02 17:08:26 +03:00
|
|
|
// Test 2: Native agent, with arguments
|
|
|
|
output = executor.execute("JVMTI.agent_load " +
|
|
|
|
libpath + " \"agent.jar=foo=bar\"");
|
2018-02-01 21:03:37 -05:00
|
|
|
checkWarningsOnly(output);
|
2016-02-18 23:26:43 +09:00
|
|
|
|
2016-03-02 17:08:26 +03:00
|
|
|
// Test 3: Java agent, no arguments
|
2016-02-18 23:26:43 +09:00
|
|
|
output = executor.execute("JVMTI.agent_load " +
|
2016-03-02 17:08:26 +03:00
|
|
|
"agent.jar");
|
2018-02-01 21:03:37 -05:00
|
|
|
checkWarningsOnly(output);
|
2016-03-02 17:08:26 +03:00
|
|
|
|
|
|
|
// Test 4: Java agent, with arguments
|
|
|
|
output = executor.execute("JVMTI.agent_load " +
|
|
|
|
"\"agent.jar=foo=bar\"");
|
2018-02-01 21:03:37 -05:00
|
|
|
checkWarningsOnly(output);
|
2016-03-02 17:08:26 +03:00
|
|
|
|
2016-02-18 23:26:43 +09:00
|
|
|
} catch (Exception e) {
|
|
|
|
throw new RuntimeException(e);
|
|
|
|
}
|
|
|
|
}
|
2016-08-19 10:06:30 -04:00
|
|
|
/**
|
|
|
|
* return path to library inside jdk tree
|
|
|
|
*/
|
|
|
|
public static String jdkLibPath() {
|
|
|
|
if (Platform.isWindows()) {
|
|
|
|
return "bin";
|
|
|
|
}
|
2016-12-20 14:15:02 +03:00
|
|
|
return "lib";
|
2016-08-19 10:06:30 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Build name of shared object according to platform rules
|
|
|
|
*/
|
|
|
|
public static String sharedObjectName(String name) {
|
|
|
|
if (Platform.isWindows()) {
|
|
|
|
return name + ".dll";
|
|
|
|
}
|
|
|
|
if (Platform.isOSX()) {
|
|
|
|
return "lib" + name + ".dylib";
|
|
|
|
}
|
|
|
|
return "lib" + name + ".so";
|
|
|
|
}
|
2016-02-18 23:26:43 +09:00
|
|
|
|
|
|
|
@Test
|
|
|
|
public void jmx() throws Throwable {
|
|
|
|
run(new JMXExecutor());
|
|
|
|
}
|
|
|
|
|
|
|
|
@Test
|
|
|
|
public void cli() throws Throwable {
|
|
|
|
run(new PidJcmdExecutor());
|
|
|
|
}
|
|
|
|
}
|