src/jdk.attach/linux/classes/sun/tools/attach/VirtualMachineImpl.java
author cjplummer
Tue, 26 Sep 2017 10:09:54 -0700
changeset 47540 7242ab15b6a5
parent 47216 71c04702a3d5
child 48485 258a4dab74a7
permissions -rw-r--r--
8179498: attach in linux should be relative to /proc/pid/root and namespace aware Summary: map pid to namespace pid so proper tmp file name is used. Reviewed-by: sspitsyn, dholmes
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     1
/*
47540
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
     2
 * Copyright (c) 2005, 2017, Oracle and/or its affiliates. All rights reserved.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
90ce3da70b43 Initial load
duke
parents:
diff changeset
     4
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
90ce3da70b43 Initial load
duke
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 2
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 2
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    10
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
90ce3da70b43 Initial load
duke
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
90ce3da70b43 Initial load
duke
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
90ce3da70b43 Initial load
duke
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
90ce3da70b43 Initial load
duke
parents:
diff changeset
    15
 * accompanied this code).
90ce3da70b43 Initial load
duke
parents:
diff changeset
    16
 *
90ce3da70b43 Initial load
duke
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
90ce3da70b43 Initial load
duke
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
90ce3da70b43 Initial load
duke
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    20
 *
5506
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 2
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 2
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
202f599c92aa 6943119: Rebrand source copyright notices
ohair
parents: 2
diff changeset
    23
 * questions.
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    24
 */
90ce3da70b43 Initial load
duke
parents:
diff changeset
    25
package sun.tools.attach;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    26
24363
33b869a8806b 8039173: Propagate errors from Diagnostic Commands as exceptions in the attach framework
sla
parents: 14342
diff changeset
    27
import com.sun.tools.attach.AttachOperationFailedException;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    28
import com.sun.tools.attach.AgentLoadException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    29
import com.sun.tools.attach.AttachNotSupportedException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    30
import com.sun.tools.attach.spi.AttachProvider;
24363
33b869a8806b 8039173: Propagate errors from Diagnostic Commands as exceptions in the attach framework
sla
parents: 14342
diff changeset
    31
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    32
import java.io.InputStream;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    33
import java.io.IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    34
import java.io.File;
47540
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
    35
import java.nio.charset.StandardCharsets;
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
    36
import java.nio.file.Path;
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
    37
import java.nio.file.Paths;
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
    38
import java.nio.file.Files;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    39
90ce3da70b43 Initial load
duke
parents:
diff changeset
    40
/*
90ce3da70b43 Initial load
duke
parents:
diff changeset
    41
 * Linux implementation of HotSpotVirtualMachine
90ce3da70b43 Initial load
duke
parents:
diff changeset
    42
 */
26216
5e46c782b43c 8055230: Rename attach provider implementation class be platform neutral
mchung
parents: 25859
diff changeset
    43
public class VirtualMachineImpl extends HotSpotVirtualMachine {
11692
dd4289e8b9e3 7132199: sun/management/jmxremote/bootstrap/JvmstatCountersTest.java failing on all platforms
sla
parents: 10419
diff changeset
    44
    // "/tmp" is used as a global well-known location for the files
dd4289e8b9e3 7132199: sun/management/jmxremote/bootstrap/JvmstatCountersTest.java failing on all platforms
sla
parents: 10419
diff changeset
    45
    // .java_pid<pid>. and .attach_pid<pid>. It is important that this
dd4289e8b9e3 7132199: sun/management/jmxremote/bootstrap/JvmstatCountersTest.java failing on all platforms
sla
parents: 10419
diff changeset
    46
    // location is the same for all processes, otherwise the tools
dd4289e8b9e3 7132199: sun/management/jmxremote/bootstrap/JvmstatCountersTest.java failing on all platforms
sla
parents: 10419
diff changeset
    47
    // will not be able to find all Hotspot processes.
dd4289e8b9e3 7132199: sun/management/jmxremote/bootstrap/JvmstatCountersTest.java failing on all platforms
sla
parents: 10419
diff changeset
    48
    // Any changes to this needs to be synchronized with HotSpot.
dd4289e8b9e3 7132199: sun/management/jmxremote/bootstrap/JvmstatCountersTest.java failing on all platforms
sla
parents: 10419
diff changeset
    49
    private static final String tmpdir = "/tmp";
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    50
90ce3da70b43 Initial load
duke
parents:
diff changeset
    51
    // The patch to the socket file created by the target VM
90ce3da70b43 Initial load
duke
parents:
diff changeset
    52
    String path;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    53
90ce3da70b43 Initial load
duke
parents:
diff changeset
    54
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
    55
     * Attaches to the target VM
90ce3da70b43 Initial load
duke
parents:
diff changeset
    56
     */
26216
5e46c782b43c 8055230: Rename attach provider implementation class be platform neutral
mchung
parents: 25859
diff changeset
    57
    VirtualMachineImpl(AttachProvider provider, String vmid)
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    58
        throws AttachNotSupportedException, IOException
90ce3da70b43 Initial load
duke
parents:
diff changeset
    59
    {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    60
        super(provider, vmid);
90ce3da70b43 Initial load
duke
parents:
diff changeset
    61
90ce3da70b43 Initial load
duke
parents:
diff changeset
    62
        // This provider only understands pids
90ce3da70b43 Initial load
duke
parents:
diff changeset
    63
        int pid;
90ce3da70b43 Initial load
duke
parents:
diff changeset
    64
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    65
            pid = Integer.parseInt(vmid);
90ce3da70b43 Initial load
duke
parents:
diff changeset
    66
        } catch (NumberFormatException x) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    67
            throw new AttachNotSupportedException("Invalid process identifier");
90ce3da70b43 Initial load
duke
parents:
diff changeset
    68
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
    69
47540
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
    70
        // Try to resolve to the "inner most" pid namespace
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
    71
        int ns_pid = getNamespacePid(pid);
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
    72
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    73
        // Find the socket file. If not found then we attempt to start the
90ce3da70b43 Initial load
duke
parents:
diff changeset
    74
        // attach mechanism in the target VM by sending it a QUIT signal.
90ce3da70b43 Initial load
duke
parents:
diff changeset
    75
        // Then we attempt to find the socket file again.
47540
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
    76
        path = findSocketFile(pid, ns_pid);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    77
        if (path == null) {
47540
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
    78
            File f = createAttachFile(pid, ns_pid);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    79
            try {
40685
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
    80
                sendQuitTo(pid);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    81
90ce3da70b43 Initial load
duke
parents:
diff changeset
    82
                // give the target VM time to start the attach mechanism
40685
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
    83
                final int delay_step = 100;
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
    84
                final long timeout = attachTimeout();
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
    85
                long time_spend = 0;
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
    86
                long delay = 0;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    87
                do {
40685
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
    88
                    // Increase timeout on each attempt to reduce polling
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
    89
                    delay += delay_step;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
    90
                    try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
    91
                        Thread.sleep(delay);
90ce3da70b43 Initial load
duke
parents:
diff changeset
    92
                    } catch (InterruptedException x) { }
47540
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
    93
                    path = findSocketFile(pid, ns_pid);
40685
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
    94
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
    95
                    time_spend += delay;
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
    96
                    if (time_spend > timeout/2 && path == null) {
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
    97
                        // Send QUIT again to give target VM the last chance to react
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
    98
                        sendQuitTo(pid);
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
    99
                    }
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
   100
                } while (time_spend <= timeout && path == null);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   101
                if (path == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   102
                    throw new AttachNotSupportedException(
40685
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
   103
                        String.format("Unable to open socket file %s: " +
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
   104
                          "target process %d doesn't respond within %dms " +
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
   105
                          "or HotSpot VM not loaded", f.getPath(), pid, time_spend));
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   106
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   107
            } finally {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   108
                f.delete();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   109
            }
40685
e6f3a9fff607 8157236: attach on ARMv7 fails with com.sun.tools.attach.AttachNotSupportedException: Unable to open socket file
dsamersoff
parents: 26216
diff changeset
   110
      }
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   111
90ce3da70b43 Initial load
duke
parents:
diff changeset
   112
        // Check that the file owner/permission to avoid attaching to
90ce3da70b43 Initial load
duke
parents:
diff changeset
   113
        // bogus process
90ce3da70b43 Initial load
duke
parents:
diff changeset
   114
        checkPermissions(path);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   115
90ce3da70b43 Initial load
duke
parents:
diff changeset
   116
        // Check that we can connect to the process
90ce3da70b43 Initial load
duke
parents:
diff changeset
   117
        // - this ensures we throw the permission denied error now rather than
90ce3da70b43 Initial load
duke
parents:
diff changeset
   118
        // later when we attempt to enqueue a command.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   119
        int s = socket();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   120
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   121
            connect(s, path);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   122
        } finally {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   123
            close(s);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   124
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   125
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   126
90ce3da70b43 Initial load
duke
parents:
diff changeset
   127
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   128
     * Detach from the target VM
90ce3da70b43 Initial load
duke
parents:
diff changeset
   129
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   130
    public void detach() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   131
        synchronized (this) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   132
            if (this.path != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   133
                this.path = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   134
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   135
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   136
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   137
90ce3da70b43 Initial load
duke
parents:
diff changeset
   138
    // protocol version
90ce3da70b43 Initial load
duke
parents:
diff changeset
   139
    private final static String PROTOCOL_VERSION = "1";
90ce3da70b43 Initial load
duke
parents:
diff changeset
   140
90ce3da70b43 Initial load
duke
parents:
diff changeset
   141
    // known errors
90ce3da70b43 Initial load
duke
parents:
diff changeset
   142
    private final static int ATTACH_ERROR_BADVERSION = 101;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   143
90ce3da70b43 Initial load
duke
parents:
diff changeset
   144
    /**
90ce3da70b43 Initial load
duke
parents:
diff changeset
   145
     * Execute the given command in the target VM.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   146
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   147
    InputStream execute(String cmd, Object ... args) throws AgentLoadException, IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   148
        assert args.length <= 3;                // includes null
90ce3da70b43 Initial load
duke
parents:
diff changeset
   149
90ce3da70b43 Initial load
duke
parents:
diff changeset
   150
        // did we detach?
90ce3da70b43 Initial load
duke
parents:
diff changeset
   151
        String p;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   152
        synchronized (this) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   153
            if (this.path == null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   154
                throw new IOException("Detached from target VM");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   155
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   156
            p = this.path;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   157
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   158
90ce3da70b43 Initial load
duke
parents:
diff changeset
   159
        // create UNIX socket
90ce3da70b43 Initial load
duke
parents:
diff changeset
   160
        int s = socket();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   161
90ce3da70b43 Initial load
duke
parents:
diff changeset
   162
        // connect to target VM
90ce3da70b43 Initial load
duke
parents:
diff changeset
   163
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   164
            connect(s, p);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   165
        } catch (IOException x) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   166
            close(s);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   167
            throw x;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   168
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   169
90ce3da70b43 Initial load
duke
parents:
diff changeset
   170
        IOException ioe = null;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   171
90ce3da70b43 Initial load
duke
parents:
diff changeset
   172
        // connected - write request
90ce3da70b43 Initial load
duke
parents:
diff changeset
   173
        // <ver> <cmd> <args...>
90ce3da70b43 Initial load
duke
parents:
diff changeset
   174
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   175
            writeString(s, PROTOCOL_VERSION);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   176
            writeString(s, cmd);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   177
90ce3da70b43 Initial load
duke
parents:
diff changeset
   178
            for (int i=0; i<3; i++) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   179
                if (i < args.length && args[i] != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   180
                    writeString(s, (String)args[i]);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   181
                } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   182
                    writeString(s, "");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   183
                }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   184
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   185
        } catch (IOException x) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   186
            ioe = x;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   187
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   188
90ce3da70b43 Initial load
duke
parents:
diff changeset
   189
90ce3da70b43 Initial load
duke
parents:
diff changeset
   190
        // Create an input stream to read reply
90ce3da70b43 Initial load
duke
parents:
diff changeset
   191
        SocketInputStream sis = new SocketInputStream(s);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   192
90ce3da70b43 Initial load
duke
parents:
diff changeset
   193
        // Read the command completion status
90ce3da70b43 Initial load
duke
parents:
diff changeset
   194
        int completionStatus;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   195
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   196
            completionStatus = readInt(sis);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   197
        } catch (IOException x) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   198
            sis.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   199
            if (ioe != null) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   200
                throw ioe;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   201
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   202
                throw x;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   203
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   204
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   205
90ce3da70b43 Initial load
duke
parents:
diff changeset
   206
        if (completionStatus != 0) {
24363
33b869a8806b 8039173: Propagate errors from Diagnostic Commands as exceptions in the attach framework
sla
parents: 14342
diff changeset
   207
            // read from the stream and use that as the error message
33b869a8806b 8039173: Propagate errors from Diagnostic Commands as exceptions in the attach framework
sla
parents: 14342
diff changeset
   208
            String message = readErrorMessage(sis);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   209
            sis.close();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   210
90ce3da70b43 Initial load
duke
parents:
diff changeset
   211
            // In the event of a protocol mismatch then the target VM
90ce3da70b43 Initial load
duke
parents:
diff changeset
   212
            // returns a known error so that we can throw a reasonable
90ce3da70b43 Initial load
duke
parents:
diff changeset
   213
            // error.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   214
            if (completionStatus == ATTACH_ERROR_BADVERSION) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   215
                throw new IOException("Protocol mismatch with target VM");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   216
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   217
90ce3da70b43 Initial load
duke
parents:
diff changeset
   218
            // Special-case the "load" command so that the right exception is
90ce3da70b43 Initial load
duke
parents:
diff changeset
   219
            // thrown.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   220
            if (cmd.equals("load")) {
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 40943
diff changeset
   221
                String msg = "Failed to load agent library";
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 40943
diff changeset
   222
                if (!message.isEmpty())
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 40943
diff changeset
   223
                    msg += ": " + message;
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 40943
diff changeset
   224
                throw new AgentLoadException(msg);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   225
            } else {
45004
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 40943
diff changeset
   226
                if (message.isEmpty())
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 40943
diff changeset
   227
                    message = "Command failed in target VM";
ea3137042a61 8178380: Module system implementation refresh (5/2017)
alanb
parents: 40943
diff changeset
   228
                throw new AttachOperationFailedException(message);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   229
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   230
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   231
90ce3da70b43 Initial load
duke
parents:
diff changeset
   232
        // Return the input stream so that the command output can be read
90ce3da70b43 Initial load
duke
parents:
diff changeset
   233
        return sis;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   234
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   235
90ce3da70b43 Initial load
duke
parents:
diff changeset
   236
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   237
     * InputStream for the socket connection to get target VM
90ce3da70b43 Initial load
duke
parents:
diff changeset
   238
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   239
    private class SocketInputStream extends InputStream {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   240
        int s;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   241
90ce3da70b43 Initial load
duke
parents:
diff changeset
   242
        public SocketInputStream(int s) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   243
            this.s = s;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   244
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   245
90ce3da70b43 Initial load
duke
parents:
diff changeset
   246
        public synchronized int read() throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   247
            byte b[] = new byte[1];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   248
            int n = this.read(b, 0, 1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   249
            if (n == 1) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   250
                return b[0] & 0xff;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   251
            } else {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   252
                return -1;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   253
            }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   254
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   255
90ce3da70b43 Initial load
duke
parents:
diff changeset
   256
        public synchronized int read(byte[] bs, int off, int len) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   257
            if ((off < 0) || (off > bs.length) || (len < 0) ||
90ce3da70b43 Initial load
duke
parents:
diff changeset
   258
                ((off + len) > bs.length) || ((off + len) < 0)) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   259
                throw new IndexOutOfBoundsException();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   260
            } else if (len == 0)
90ce3da70b43 Initial load
duke
parents:
diff changeset
   261
                return 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   262
26216
5e46c782b43c 8055230: Rename attach provider implementation class be platform neutral
mchung
parents: 25859
diff changeset
   263
            return VirtualMachineImpl.read(s, bs, off, len);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   264
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   265
90ce3da70b43 Initial load
duke
parents:
diff changeset
   266
        public void close() throws IOException {
26216
5e46c782b43c 8055230: Rename attach provider implementation class be platform neutral
mchung
parents: 25859
diff changeset
   267
            VirtualMachineImpl.close(s);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   268
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   269
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   270
90ce3da70b43 Initial load
duke
parents:
diff changeset
   271
    // Return the socket file for the given process.
47540
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   272
    private String findSocketFile(int pid, int ns_pid) {
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   273
        // A process may not exist in the same mount namespace as the caller.
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   274
        // Instead, attach relative to the target root filesystem as exposed by
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   275
        // procfs regardless of namespaces.
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   276
        String root = "/proc/" + pid + "/root/" + tmpdir;
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   277
        File f = new File(root, ".java_pid" + ns_pid);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   278
        if (!f.exists()) {
11692
dd4289e8b9e3 7132199: sun/management/jmxremote/bootstrap/JvmstatCountersTest.java failing on all platforms
sla
parents: 10419
diff changeset
   279
            return null;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   280
        }
11692
dd4289e8b9e3 7132199: sun/management/jmxremote/bootstrap/JvmstatCountersTest.java failing on all platforms
sla
parents: 10419
diff changeset
   281
        return f.getPath();
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   282
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   283
90ce3da70b43 Initial load
duke
parents:
diff changeset
   284
    // On Solaris/Linux a simple handshake is used to start the attach mechanism
90ce3da70b43 Initial load
duke
parents:
diff changeset
   285
    // if not already started. The client creates a .attach_pid<pid> file in the
5771
1f415c7e277d 6950927: Testcase failure sun/management/jmxremote/bootstrap/JvmstatCountersTest.java
alanb
parents: 5506
diff changeset
   286
    // target VM's working directory (or temp directory), and the SIGQUIT handler
1f415c7e277d 6950927: Testcase failure sun/management/jmxremote/bootstrap/JvmstatCountersTest.java
alanb
parents: 5506
diff changeset
   287
    // checks for the file.
47540
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   288
    private File createAttachFile(int pid, int ns_pid) throws IOException {
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   289
        String fn = ".attach_pid" + ns_pid;
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   290
        String path = "/proc/" + pid + "/cwd/" + fn;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   291
        File f = new File(path);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   292
        try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   293
            f.createNewFile();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   294
        } catch (IOException x) {
47540
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   295
            String root;
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   296
            if (pid != ns_pid) {
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   297
                // A process may not exist in the same mount namespace as the caller.
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   298
                // Instead, attach relative to the target root filesystem as exposed by
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   299
                // procfs regardless of namespaces.
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   300
                root = "/proc/" + pid + "/root/" + tmpdir;
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   301
            } else {
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   302
                root = tmpdir;
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   303
            }
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   304
            f = new File(root, fn);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   305
            f.createNewFile();
90ce3da70b43 Initial load
duke
parents:
diff changeset
   306
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   307
        return f;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   308
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   309
90ce3da70b43 Initial load
duke
parents:
diff changeset
   310
    /*
90ce3da70b43 Initial load
duke
parents:
diff changeset
   311
     * Write/sends the given to the target VM. String is transmitted in
90ce3da70b43 Initial load
duke
parents:
diff changeset
   312
     * UTF-8 encoding.
90ce3da70b43 Initial load
duke
parents:
diff changeset
   313
     */
90ce3da70b43 Initial load
duke
parents:
diff changeset
   314
    private void writeString(int fd, String s) throws IOException {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   315
        if (s.length() > 0) {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   316
            byte b[];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   317
            try {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   318
                b = s.getBytes("UTF-8");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   319
            } catch (java.io.UnsupportedEncodingException x) {
10419
12c063b39232 7084245: Update usages of InternalError to use exception chaining
sherman
parents: 7668
diff changeset
   320
                throw new InternalError(x);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   321
            }
26216
5e46c782b43c 8055230: Rename attach provider implementation class be platform neutral
mchung
parents: 25859
diff changeset
   322
            VirtualMachineImpl.write(fd, b, 0, b.length);
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   323
        }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   324
        byte b[] = new byte[1];
90ce3da70b43 Initial load
duke
parents:
diff changeset
   325
        b[0] = 0;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   326
        write(fd, b, 0, 1);
90ce3da70b43 Initial load
duke
parents:
diff changeset
   327
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   328
90ce3da70b43 Initial load
duke
parents:
diff changeset
   329
47540
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   330
    // Return the inner most namespaced PID if there is one,
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   331
    // otherwise return the original PID.
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   332
    private int getNamespacePid(int pid) throws AttachNotSupportedException, IOException {
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   333
        // Assuming a real procfs sits beneath, reading this doesn't block
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   334
        // nor will it consume a lot of memory.
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   335
        String statusFile = "/proc/" + pid + "/status";
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   336
        File f = new File(statusFile);
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   337
        if (!f.exists()) {
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   338
            return pid; // Likely a bad pid, but this is properly handled later.
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   339
        }
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   340
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   341
        Path statusPath = Paths.get(statusFile);
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   342
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   343
        try {
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   344
            for (String line : Files.readAllLines(statusPath, StandardCharsets.UTF_8)) {
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   345
                String[] parts = line.split(":");
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   346
                if (parts.length == 2 && parts[0].trim().equals("NSpid")) {
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   347
                    parts = parts[1].trim().split("\\s+");
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   348
                    // The last entry represents the PID the JVM "thinks" it is.
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   349
                    // Even in non-namespaced pids these entries should be
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   350
                    // valid. You could refer to it as the inner most pid.
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   351
                    int ns_pid = Integer.parseInt(parts[parts.length - 1]);
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   352
                    return ns_pid;
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   353
                }
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   354
            }
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   355
            // Old kernels may not have NSpid field (i.e. 3.10).
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   356
            // Fallback to original pid in the event we cannot deduce.
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   357
            return pid;
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   358
        } catch (NumberFormatException | IOException x) {
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   359
            throw new AttachNotSupportedException("Unable to parse namespace");
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   360
        }
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   361
    }
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   362
7242ab15b6a5 8179498: attach in linux should be relative to /proc/pid/root and namespace aware
cjplummer
parents: 47216
diff changeset
   363
2
90ce3da70b43 Initial load
duke
parents:
diff changeset
   364
    //-- native methods
90ce3da70b43 Initial load
duke
parents:
diff changeset
   365
90ce3da70b43 Initial load
duke
parents:
diff changeset
   366
    static native void sendQuitToChildrenOf(int pid) throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   367
90ce3da70b43 Initial load
duke
parents:
diff changeset
   368
    static native void sendQuitTo(int pid) throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   369
90ce3da70b43 Initial load
duke
parents:
diff changeset
   370
    static native void checkPermissions(String path) throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   371
90ce3da70b43 Initial load
duke
parents:
diff changeset
   372
    static native int socket() throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   373
90ce3da70b43 Initial load
duke
parents:
diff changeset
   374
    static native void connect(int fd, String path) throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   375
90ce3da70b43 Initial load
duke
parents:
diff changeset
   376
    static native void close(int fd) throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   377
90ce3da70b43 Initial load
duke
parents:
diff changeset
   378
    static native int read(int fd, byte buf[], int off, int bufLen) throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   379
90ce3da70b43 Initial load
duke
parents:
diff changeset
   380
    static native void write(int fd, byte buf[], int off, int bufLen) throws IOException;
90ce3da70b43 Initial load
duke
parents:
diff changeset
   381
90ce3da70b43 Initial load
duke
parents:
diff changeset
   382
    static {
90ce3da70b43 Initial load
duke
parents:
diff changeset
   383
        System.loadLibrary("attach");
90ce3da70b43 Initial load
duke
parents:
diff changeset
   384
    }
90ce3da70b43 Initial load
duke
parents:
diff changeset
   385
}