src/jdk.jfr/share/classes/jdk/jfr/internal/dcmd/DCmdStart.java
author mgronlun
Wed, 30 Oct 2019 19:43:52 +0100
changeset 58863 c16ac7a2eba4
parent 55302 686dedba1d9a
permissions -rw-r--r--
8226511: Implement JFR Event Streaming Reviewed-by: egahlin, mseledtsov, mgronlun Contributed-by: erik.gahlin@oracle.com, mikhailo.seledtsov@oracle.com, markus.gronlund@oracle.com
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     1
/*
58863
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
     2
 * Copyright (c) 2012, 2019, Oracle and/or its affiliates. All rights reserved.
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     4
 *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    10
 *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    15
 * accompanied this code).
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    16
 *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    20
 *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    23
 * questions.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    24
 */
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    25
package jdk.jfr.internal.dcmd;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    26
55302
686dedba1d9a 8216064: -XX:StartFlightRecording:settings= doesn't work properly
egahlin
parents: 52413
diff changeset
    27
import java.io.FileNotFoundException;
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    28
import java.io.IOException;
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    29
import java.nio.file.Files;
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    30
import java.nio.file.InvalidPathException;
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    31
import java.nio.file.Path;
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    32
import java.nio.file.Paths;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    33
import java.text.ParseException;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    34
import java.time.Duration;
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    35
import java.util.Arrays;
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    36
import java.util.HashMap;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    37
import java.util.Map;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    38
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    39
import jdk.jfr.FlightRecorder;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    40
import jdk.jfr.Recording;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    41
import jdk.jfr.internal.JVM;
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    42
import jdk.jfr.internal.LogLevel;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    43
import jdk.jfr.internal.LogTag;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    44
import jdk.jfr.internal.Logger;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    45
import jdk.jfr.internal.OldObjectSample;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    46
import jdk.jfr.internal.PrivateAccess;
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    47
import jdk.jfr.internal.SecuritySupport.SafePath;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    48
import jdk.jfr.internal.Type;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    49
import jdk.jfr.internal.jfc.JFC;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    50
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    51
/**
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    52
 * JFR.start
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    53
 *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    54
 */
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    55
//Instantiated by native
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    56
final class DCmdStart extends AbstractDCmd {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    57
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    58
    /**
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    59
     * Execute JFR.start.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    60
     *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    61
     * @param name optional name that can be used to identify recording.
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    62
     * @param settings names of settings files to use, i.e. "default" or
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    63
     *        "default.jfc".
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    64
     * @param delay delay before recording is started, in nanoseconds. Must be
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    65
     *        at least 1 second.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    66
     * @param duration duration of the recording, in nanoseconds. Must be at
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    67
     *        least 1 second.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    68
     * @param disk if recording should be persisted to disk
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    69
     * @param path file path where recording data should be written
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    70
     * @param maxAge how long recording data should be kept in the disk
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    71
     *        repository, or <code>0</code> if no limit should be set.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    72
     *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    73
     * @param maxSize the minimum amount data to keep in the disk repository
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    74
     *        before it is discarded, or <code>0</code> if no limit should be
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    75
     *        set.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    76
     *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    77
     * @param dumpOnExit if recording should dump on exit
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    78
     *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    79
     * @return result output
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    80
     *
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    81
     * @throws DCmdException if recording could not be started
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    82
     */
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    83
    @SuppressWarnings("resource")
58863
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
    84
    public String execute(String name, String[] settings, Long delay, Long duration, Boolean disk, String path, Long maxAge, Long maxSize, Long flush, Boolean dumpOnExit, Boolean pathToGcRoots) throws DCmdException {
52413
6372f5af9612 8209960: -Xlog:jfr* doesn't work with the JFR
egahlin
parents: 50887
diff changeset
    85
        if (Logger.shouldLog(LogTag.JFR_DCMD, LogLevel.DEBUG)) {
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    86
            Logger.log(LogTag.JFR_DCMD, LogLevel.DEBUG, "Executing DCmdStart: name=" + name +
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    87
                    ", settings=" + Arrays.asList(settings) +
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    88
                    ", delay=" + delay +
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    89
                    ", duration=" + duration +
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    90
                    ", disk=" + disk+
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    91
                    ", filename=" + path +
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    92
                    ", maxage=" + maxAge +
58863
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
    93
                    ", flush=" + flush +
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    94
                    ", maxsize=" + maxSize +
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    95
                    ", dumponexit =" + dumpOnExit +
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    96
                    ", path-to-gc-roots=" + pathToGcRoots);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
    97
        }
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    98
        if (name != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    99
            try {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   100
                Integer.parseInt(name);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   101
                throw new DCmdException("Name of recording can't be numeric");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   102
            } catch (NumberFormatException nfe) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   103
                // ok, can't be mixed up with name
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   104
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   105
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   106
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   107
        if (duration == null && Boolean.FALSE.equals(dumpOnExit) && path != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   108
            throw new DCmdException("Filename can only be set for a time bound recording or if dumponexit=true. Set duration/dumponexit or omit filename.");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   109
        }
55302
686dedba1d9a 8216064: -XX:StartFlightRecording:settings= doesn't work properly
egahlin
parents: 52413
diff changeset
   110
        if (settings.length == 1 && settings[0].length() == 0) {
686dedba1d9a 8216064: -XX:StartFlightRecording:settings= doesn't work properly
egahlin
parents: 52413
diff changeset
   111
            throw new DCmdException("No settings specified. Use settings=none to start without any settings");
686dedba1d9a 8216064: -XX:StartFlightRecording:settings= doesn't work properly
egahlin
parents: 52413
diff changeset
   112
        }
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   113
        Map<String, String> s = new HashMap<>();
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   114
        for (String configName : settings) {
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   115
            try {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   116
                s.putAll(JFC.createKnown(configName).getSettings());
55302
686dedba1d9a 8216064: -XX:StartFlightRecording:settings= doesn't work properly
egahlin
parents: 52413
diff changeset
   117
            } catch(FileNotFoundException e) {
686dedba1d9a 8216064: -XX:StartFlightRecording:settings= doesn't work properly
egahlin
parents: 52413
diff changeset
   118
                throw new DCmdException("Could not find settings file'" + configName + "'", e);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   119
            } catch (IOException | ParseException e) {
55302
686dedba1d9a 8216064: -XX:StartFlightRecording:settings= doesn't work properly
egahlin
parents: 52413
diff changeset
   120
                throw new DCmdException("Could not parse settings file '" + settings[0] + "'", e);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   121
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   122
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   123
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   124
        OldObjectSample.updateSettingPathToGcRoots(s, pathToGcRoots);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   125
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   126
        if (duration != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   127
            if (duration < 1000L * 1000L * 1000L) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   128
                // to avoid typo, duration below 1s makes no sense
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   129
                throw new DCmdException("Could not start recording, duration must be at least 1 second.");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   130
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   131
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   132
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   133
        if (delay != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   134
            if (delay < 1000L * 1000L * 1000) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   135
                // to avoid typo, delay shorter than 1s makes no sense.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   136
                throw new DCmdException("Could not start recording, delay must be at least 1 second.");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   137
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   138
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   139
58863
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
   140
        if (flush != null) {
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
   141
            if (Boolean.FALSE.equals(disk)) {
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
   142
                throw new DCmdException("Flush can only be set for recordings that are to disk.");
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
   143
            }
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
   144
        }
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
   145
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   146
        if (!FlightRecorder.isInitialized() && delay == null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   147
            initializeWithForcedInstrumentation(s);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   148
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   149
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   150
        Recording recording = new Recording();
50226
408021edf22f 8195819: Remove recording=x from jcmd JFR.check output
egahlin
parents: 50113
diff changeset
   151
        if (name != null) {
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   152
            recording.setName(name);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   153
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   154
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   155
        if (disk != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   156
            recording.setToDisk(disk.booleanValue());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   157
        }
58863
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
   158
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   159
        recording.setSettings(s);
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   160
        SafePath safePath = null;
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   161
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   162
        if (path != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   163
            try {
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   164
                if (dumpOnExit == null) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   165
                    // default to dumponexit=true if user specified filename
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   166
                    dumpOnExit = Boolean.TRUE;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   167
                }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   168
                Path p = Paths.get(path);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   169
                if (Files.isDirectory(p) && Boolean.TRUE.equals(dumpOnExit)) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   170
                    // Decide destination filename at dump time
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   171
                    // Purposely avoid generating filename in Recording#setDestination due to
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   172
                    // security concerns
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   173
                    PrivateAccess.getInstance().getPlatformRecording(recording).setDumpOnExitDirectory(new SafePath(p));
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   174
                } else {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   175
                    safePath = resolvePath(recording, path);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   176
                    recording.setDestination(safePath.toPath());
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   177
                }
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   178
            } catch (IOException | InvalidPathException e) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   179
                recording.close();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   180
                throw new DCmdException("Could not start recording, not able to write to file %s. %s ", path, e.getMessage());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   181
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   182
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   183
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   184
        if (maxAge != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   185
            recording.setMaxAge(Duration.ofNanos(maxAge));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   186
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   187
58863
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
   188
        if (flush != null) {
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
   189
            recording.setFlushInterval(Duration.ofNanos(flush));
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
   190
        }
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
   191
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   192
        if (maxSize != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   193
            recording.setMaxSize(maxSize);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   194
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   195
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   196
        if (duration != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   197
            recording.setDuration(Duration.ofNanos(duration));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   198
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   199
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   200
        if (dumpOnExit != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   201
            recording.setDumpOnExit(dumpOnExit);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   202
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   203
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   204
        if (delay != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   205
            Duration dDelay = Duration.ofNanos(delay);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   206
            recording.scheduleStart(dDelay);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   207
            print("Recording " + recording.getId() + " scheduled to start in ");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   208
            printTimespan(dDelay, " ");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   209
            print(".");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   210
        } else {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   211
            recording.start();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   212
            print("Started recording " + recording.getId() + ".");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   213
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   214
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   215
        if (recording.isToDisk() && duration == null && maxAge == null && maxSize == null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   216
            print(" No limit specified, using maxsize=250MB as default.");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   217
            recording.setMaxSize(250*1024L*1024L);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   218
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   219
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   220
        if (safePath != null && duration != null) {
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   221
            println(" The result will be written to:");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   222
            println();
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50226
diff changeset
   223
            printPath(safePath);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   224
        } else {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   225
            println();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   226
            println();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   227
            String cmd = duration == null ? "dump" : "stop";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   228
            String fileOption = path == null ? "filename=FILEPATH " : "";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   229
            String recordingspecifier = "name=" + recording.getId();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   230
            // if user supplied a name, use it.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   231
            if (name != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   232
                recordingspecifier = "name=" + quoteIfNeeded(name);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   233
            }
50887
fa91003ec489 8137164: Include PID in the JFR jcmd hint
egahlin
parents: 50886
diff changeset
   234
            print("Use jcmd " + getPid() + " JFR." + cmd + " " + recordingspecifier + " " + fileOption + "to copy recording data to file.");
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   235
            println();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   236
        }
58863
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 55302
diff changeset
   237
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   238
        return getResult();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   239
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   240
50887
fa91003ec489 8137164: Include PID in the JFR jcmd hint
egahlin
parents: 50886
diff changeset
   241
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   242
    // Instruments JDK-events on class load to reduce startup time
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   243
    private void initializeWithForcedInstrumentation(Map<String, String> settings) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   244
        if (!hasJDKEvents(settings)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   245
            return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   246
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   247
        JVM jvm = JVM.getJVM();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   248
        try {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   249
            jvm.setForceInstrumentation(true);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   250
            FlightRecorder.getFlightRecorder();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   251
        } finally {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   252
            jvm.setForceInstrumentation(false);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   253
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   254
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   255
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   256
    private boolean hasJDKEvents(Map<String, String> settings) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   257
        String[] eventNames = new String[7];
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   258
        eventNames[0] = "FileRead";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   259
        eventNames[1] = "FileWrite";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   260
        eventNames[2] = "SocketRead";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   261
        eventNames[3] = "SocketWrite";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   262
        eventNames[4] = "JavaErrorThrow";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   263
        eventNames[5] = "JavaExceptionThrow";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   264
        eventNames[6] = "FileForce";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   265
        for (String eventName : eventNames) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   266
            if ("true".equals(settings.get(Type.EVENT_NAME_PREFIX + eventName + "#enabled"))) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   267
                return true;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   268
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   269
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   270
        return false;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   271
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   272
}