mirror of
https://github.com/openjdk/jdk.git
synced 2025-08-28 07:14:30 +02:00
395 lines
14 KiB
Java
395 lines
14 KiB
Java
/*
|
|
* Copyright (c) 2005, 2018, 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. Oracle designates this
|
|
* particular file as subject to the "Classpath" exception as provided
|
|
* by Oracle in the LICENSE file that accompanied this code.
|
|
*
|
|
* 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.
|
|
*/
|
|
|
|
package sun.tools.attach;
|
|
|
|
import com.sun.tools.attach.AttachNotSupportedException;
|
|
import com.sun.tools.attach.VirtualMachine;
|
|
import com.sun.tools.attach.AgentLoadException;
|
|
import com.sun.tools.attach.AgentInitializationException;
|
|
import com.sun.tools.attach.spi.AttachProvider;
|
|
import jdk.internal.misc.VM;
|
|
|
|
import java.io.BufferedReader;
|
|
import java.io.InputStream;
|
|
import java.io.IOException;
|
|
import java.io.InputStreamReader;
|
|
import java.security.AccessController;
|
|
import java.security.PrivilegedAction;
|
|
import java.util.Properties;
|
|
import java.util.stream.Collectors;
|
|
|
|
/*
|
|
* The HotSpot implementation of com.sun.tools.attach.VirtualMachine.
|
|
*/
|
|
|
|
public abstract class HotSpotVirtualMachine extends VirtualMachine {
|
|
|
|
private static final long CURRENT_PID;
|
|
private static final boolean ALLOW_ATTACH_SELF;
|
|
static {
|
|
PrivilegedAction<ProcessHandle> pa = ProcessHandle::current;
|
|
CURRENT_PID = AccessController.doPrivileged(pa).pid();
|
|
|
|
String s = VM.getSavedProperty("jdk.attach.allowAttachSelf");
|
|
ALLOW_ATTACH_SELF = "".equals(s) || Boolean.parseBoolean(s);
|
|
}
|
|
|
|
HotSpotVirtualMachine(AttachProvider provider, String id)
|
|
throws AttachNotSupportedException, IOException
|
|
{
|
|
super(provider, id);
|
|
|
|
int pid;
|
|
try {
|
|
pid = Integer.parseInt(id);
|
|
} catch (NumberFormatException e) {
|
|
throw new AttachNotSupportedException("Invalid process identifier");
|
|
}
|
|
|
|
// The tool should be a different VM to the target. This check will
|
|
// eventually be enforced by the target VM.
|
|
if (!ALLOW_ATTACH_SELF && (pid == 0 || pid == CURRENT_PID)) {
|
|
throw new IOException("Can not attach to current VM");
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Load agent library
|
|
* If isAbsolute is true then the agent library is the absolute path
|
|
* to the library and thus will not be expanded in the target VM.
|
|
* if isAbsolute is false then the agent library is just a library
|
|
* name and it will be expended in the target VM.
|
|
*/
|
|
private void loadAgentLibrary(String agentLibrary, boolean isAbsolute, String options)
|
|
throws AgentLoadException, AgentInitializationException, IOException
|
|
{
|
|
if (agentLibrary == null) {
|
|
throw new NullPointerException("agentLibrary cannot be null");
|
|
}
|
|
|
|
String msgPrefix = "return code: ";
|
|
InputStream in = execute("load",
|
|
agentLibrary,
|
|
isAbsolute ? "true" : "false",
|
|
options);
|
|
try (BufferedReader reader = new BufferedReader(new InputStreamReader(in))) {
|
|
String result = reader.readLine();
|
|
if (result == null) {
|
|
throw new AgentLoadException("Target VM did not respond");
|
|
} else if (result.startsWith(msgPrefix)) {
|
|
int retCode = Integer.parseInt(result.substring(msgPrefix.length()));
|
|
if (retCode != 0) {
|
|
throw new AgentInitializationException("Agent_OnAttach failed", retCode);
|
|
}
|
|
} else {
|
|
throw new AgentLoadException(result);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Load agent library - library name will be expanded in target VM
|
|
*/
|
|
public void loadAgentLibrary(String agentLibrary, String options)
|
|
throws AgentLoadException, AgentInitializationException, IOException
|
|
{
|
|
loadAgentLibrary(agentLibrary, false, options);
|
|
}
|
|
|
|
/*
|
|
* Load agent - absolute path of library provided to target VM
|
|
*/
|
|
public void loadAgentPath(String agentLibrary, String options)
|
|
throws AgentLoadException, AgentInitializationException, IOException
|
|
{
|
|
loadAgentLibrary(agentLibrary, true, options);
|
|
}
|
|
|
|
/*
|
|
* Load JPLIS agent which will load the agent JAR file and invoke
|
|
* the agentmain method.
|
|
*/
|
|
public void loadAgent(String agent, String options)
|
|
throws AgentLoadException, AgentInitializationException, IOException
|
|
{
|
|
if (agent == null) {
|
|
throw new NullPointerException("agent cannot be null");
|
|
}
|
|
|
|
String args = agent;
|
|
if (options != null) {
|
|
args = args + "=" + options;
|
|
}
|
|
try {
|
|
loadAgentLibrary("instrument", args);
|
|
} catch (AgentInitializationException x) {
|
|
/*
|
|
* Translate interesting errors into the right exception and
|
|
* message (FIXME: create a better interface to the instrument
|
|
* implementation so this isn't necessary)
|
|
*/
|
|
int rc = x.returnValue();
|
|
switch (rc) {
|
|
case JNI_ENOMEM:
|
|
throw new AgentLoadException("Insuffient memory");
|
|
case ATTACH_ERROR_BADJAR:
|
|
throw new AgentLoadException(
|
|
"Agent JAR not found or no Agent-Class attribute");
|
|
case ATTACH_ERROR_NOTONCP:
|
|
throw new AgentLoadException(
|
|
"Unable to add JAR file to system class path");
|
|
case ATTACH_ERROR_STARTFAIL:
|
|
throw new AgentInitializationException(
|
|
"Agent JAR loaded but agent failed to initialize");
|
|
default :
|
|
throw new AgentLoadException("" +
|
|
"Failed to load agent - unknown reason: " + rc);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
* The possible errors returned by JPLIS's agentmain
|
|
*/
|
|
private static final int JNI_ENOMEM = -4;
|
|
private static final int ATTACH_ERROR_BADJAR = 100;
|
|
private static final int ATTACH_ERROR_NOTONCP = 101;
|
|
private static final int ATTACH_ERROR_STARTFAIL = 102;
|
|
|
|
|
|
/*
|
|
* Send "properties" command to target VM
|
|
*/
|
|
public Properties getSystemProperties() throws IOException {
|
|
InputStream in = null;
|
|
Properties props = new Properties();
|
|
try {
|
|
in = executeCommand("properties");
|
|
props.load(in);
|
|
} finally {
|
|
if (in != null) in.close();
|
|
}
|
|
return props;
|
|
}
|
|
|
|
public Properties getAgentProperties() throws IOException {
|
|
InputStream in = null;
|
|
Properties props = new Properties();
|
|
try {
|
|
in = executeCommand("agentProperties");
|
|
props.load(in);
|
|
} finally {
|
|
if (in != null) in.close();
|
|
}
|
|
return props;
|
|
}
|
|
|
|
private static final String MANAGEMENT_PREFIX = "com.sun.management.";
|
|
|
|
private static boolean checkedKeyName(Object key) {
|
|
if (!(key instanceof String)) {
|
|
throw new IllegalArgumentException("Invalid option (not a String): "+key);
|
|
}
|
|
if (!((String)key).startsWith(MANAGEMENT_PREFIX)) {
|
|
throw new IllegalArgumentException("Invalid option: "+key);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
private static String stripKeyName(Object key) {
|
|
return ((String)key).substring(MANAGEMENT_PREFIX.length());
|
|
}
|
|
|
|
@Override
|
|
public void startManagementAgent(Properties agentProperties) throws IOException {
|
|
if (agentProperties == null) {
|
|
throw new NullPointerException("agentProperties cannot be null");
|
|
}
|
|
// Convert the arguments into arguments suitable for the Diagnostic Command:
|
|
// "ManagementAgent.start jmxremote.port=5555 jmxremote.authenticate=false"
|
|
String args = agentProperties.entrySet().stream()
|
|
.filter(entry -> checkedKeyName(entry.getKey()))
|
|
.map(entry -> stripKeyName(entry.getKey()) + "=" + escape(entry.getValue()))
|
|
.collect(Collectors.joining(" "));
|
|
executeJCmd("ManagementAgent.start " + args).close();
|
|
}
|
|
|
|
private String escape(Object arg) {
|
|
String value = arg.toString();
|
|
if (value.contains(" ")) {
|
|
return "'" + value + "'";
|
|
}
|
|
return value;
|
|
}
|
|
|
|
@Override
|
|
public String startLocalManagementAgent() throws IOException {
|
|
executeJCmd("ManagementAgent.start_local").close();
|
|
String prop = MANAGEMENT_PREFIX + "jmxremote.localConnectorAddress";
|
|
return getAgentProperties().getProperty(prop);
|
|
}
|
|
|
|
|
|
// --- HotSpot specific methods ---
|
|
|
|
// same as SIGQUIT
|
|
public void localDataDump() throws IOException {
|
|
executeCommand("datadump").close();
|
|
}
|
|
|
|
// Remote ctrl-break. The output of the ctrl-break actions can
|
|
// be read from the input stream.
|
|
public InputStream remoteDataDump(Object ... args) throws IOException {
|
|
return executeCommand("threaddump", args);
|
|
}
|
|
|
|
// Remote heap dump. The output (error message) can be read from the
|
|
// returned input stream.
|
|
public InputStream dumpHeap(Object ... args) throws IOException {
|
|
return executeCommand("dumpheap", args);
|
|
}
|
|
|
|
// Heap histogram (heap inspection in HotSpot)
|
|
public InputStream heapHisto(Object ... args) throws IOException {
|
|
return executeCommand("inspectheap", args);
|
|
}
|
|
|
|
// set JVM command line flag
|
|
public InputStream setFlag(String name, String value) throws IOException {
|
|
return executeCommand("setflag", name, value);
|
|
}
|
|
|
|
// print command line flag
|
|
public InputStream printFlag(String name) throws IOException {
|
|
return executeCommand("printflag", name);
|
|
}
|
|
|
|
public InputStream executeJCmd(String command) throws IOException {
|
|
return executeCommand("jcmd", command);
|
|
}
|
|
|
|
|
|
// -- Supporting methods
|
|
|
|
/*
|
|
* Execute the given command in the target VM - specific platform
|
|
* implementation must implement this.
|
|
*/
|
|
abstract InputStream execute(String cmd, Object ... args)
|
|
throws AgentLoadException, IOException;
|
|
|
|
/*
|
|
* Convenience method for simple commands
|
|
*/
|
|
public InputStream executeCommand(String cmd, Object ... args) throws IOException {
|
|
try {
|
|
return execute(cmd, args);
|
|
} catch (AgentLoadException x) {
|
|
throw new InternalError("Should not get here", x);
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
* Utility method to read an 'int' from the input stream. Ideally
|
|
* we should be using java.util.Scanner here but this implementation
|
|
* guarantees not to read ahead.
|
|
*/
|
|
int readInt(InputStream in) throws IOException {
|
|
StringBuilder sb = new StringBuilder();
|
|
|
|
// read to \n or EOF
|
|
int n;
|
|
byte buf[] = new byte[1];
|
|
do {
|
|
n = in.read(buf, 0, 1);
|
|
if (n > 0) {
|
|
char c = (char)buf[0];
|
|
if (c == '\n') {
|
|
break; // EOL found
|
|
} else {
|
|
sb.append(c);
|
|
}
|
|
}
|
|
} while (n > 0);
|
|
|
|
if (sb.length() == 0) {
|
|
throw new IOException("Premature EOF");
|
|
}
|
|
|
|
int value;
|
|
try {
|
|
value = Integer.parseInt(sb.toString());
|
|
} catch (NumberFormatException x) {
|
|
throw new IOException("Non-numeric value found - int expected");
|
|
}
|
|
return value;
|
|
}
|
|
|
|
/*
|
|
* Utility method to read data into a String.
|
|
*/
|
|
String readErrorMessage(InputStream in) throws IOException {
|
|
String s;
|
|
StringBuilder message = new StringBuilder();
|
|
BufferedReader br = new BufferedReader(new InputStreamReader(in));
|
|
while ((s = br.readLine()) != null) {
|
|
message.append(s);
|
|
}
|
|
return message.toString();
|
|
}
|
|
|
|
|
|
// -- attach timeout support
|
|
|
|
private static long defaultAttachTimeout = 10000;
|
|
private volatile long attachTimeout;
|
|
|
|
/*
|
|
* Return attach timeout based on the value of the sun.tools.attach.attachTimeout
|
|
* property, or the default timeout if the property is not set to a positive
|
|
* value.
|
|
*/
|
|
long attachTimeout() {
|
|
if (attachTimeout == 0) {
|
|
synchronized(this) {
|
|
if (attachTimeout == 0) {
|
|
try {
|
|
String s =
|
|
System.getProperty("sun.tools.attach.attachTimeout");
|
|
attachTimeout = Long.parseLong(s);
|
|
} catch (SecurityException se) {
|
|
} catch (NumberFormatException ne) {
|
|
}
|
|
if (attachTimeout <= 0) {
|
|
attachTimeout = defaultAttachTimeout;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return attachTimeout;
|
|
}
|
|
}
|