src/jdk.jfr/share/classes/jdk/jfr/internal/PlatformRecording.java
author egahlin
Mon, 21 May 2018 18:19:11 +0200
changeset 50197 f4735ff8d17d
parent 50194 2ae4cd9d809d
child 50745 a390cbb82d47
permissions -rw-r--r--
8195818: JFR.start should increase autogenerated name by one Reviewed-by: mgronlun
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     1
/*
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
     2
 * Copyright (c) 2016, 2018, Oracle and/or its affiliates. All rights reserved.
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
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    26
package jdk.jfr.internal;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    27
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    28
import static jdk.jfr.internal.LogLevel.DEBUG;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    29
import static jdk.jfr.internal.LogLevel.WARN;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    30
import static jdk.jfr.internal.LogTag.JFR;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    31
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    32
import java.io.IOException;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    33
import java.io.InputStream;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    34
import java.nio.channels.FileChannel;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    35
import java.nio.file.StandardOpenOption;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    36
import java.security.AccessControlContext;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    37
import java.security.AccessController;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    38
import java.time.Duration;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    39
import java.time.Instant;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    40
import java.time.LocalDateTime;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    41
import java.util.ArrayList;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    42
import java.util.Collections;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    43
import java.util.Date;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    44
import java.util.HashMap;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    45
import java.util.LinkedHashMap;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    46
import java.util.LinkedList;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    47
import java.util.List;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    48
import java.util.Map;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    49
import java.util.StringJoiner;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    50
import java.util.TimerTask;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    51
import java.util.TreeMap;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    52
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    53
import jdk.jfr.Configuration;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    54
import jdk.jfr.FlightRecorderListener;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    55
import jdk.jfr.Recording;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    56
import jdk.jfr.RecordingState;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    57
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    58
public final class PlatformRecording implements AutoCloseable {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    59
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    60
    private final PlatformRecorder recorder;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    61
    private final long id;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    62
    // Recording settings
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    63
    private Map<String, String> settings = new LinkedHashMap<>();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    64
    private Duration duration;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    65
    private Duration maxAge;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    66
    private long maxSize;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    67
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    68
    private WriteableUserPath destination;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    69
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    70
    private boolean toDisk = true;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    71
    private String name;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    72
    private boolean dumpOnExit;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    73
    // Timestamp information
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    74
    private Instant stopTime;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    75
    private Instant startTime;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    76
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    77
    // Misc, information
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    78
    private RecordingState state = RecordingState.NEW;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    79
    private long size;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    80
    private final LinkedList<RepositoryChunk> chunks = new LinkedList<>();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    81
    private volatile Recording recording;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    82
    private TimerTask stopTask;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    83
    private TimerTask startTask;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    84
    private AccessControlContext noDestinationDumpOnExitAccessControlContext;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    85
    private boolean shuoldWriteActiveRecordingEvent = true;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    86
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    87
    PlatformRecording(PlatformRecorder recorder, long id) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    88
        // Typically the access control context is taken
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    89
        // when you call dump(Path) or setDdestination(Path),
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    90
        // but if no destination is set and dumponexit=true
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    91
        // the control context of the recording is taken when the
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    92
        // Recording object is constructed.  This works well for
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    93
        // -XX:StartFlightRecording and JFR.dump
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    94
        this.noDestinationDumpOnExitAccessControlContext = AccessController.getContext();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    95
        this.id = id;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    96
        this.recorder = recorder;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    97
        this.name = String.valueOf(id);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    98
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    99
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   100
    public void start() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   101
        RecordingState oldState;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   102
        RecordingState newState;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   103
        synchronized (recorder) {
50197
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   104
            oldState = getState();
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   105
            if (!Utils.isBefore(state, RecordingState.RUNNING)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   106
                throw new IllegalStateException("Recording can only be started once.");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   107
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   108
            if (startTask != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   109
                startTask.cancel();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   110
                startTask = null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   111
                startTime = null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   112
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   113
            recorder.start(this);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   114
            Logger.log(LogTag.JFR, LogLevel.INFO, () -> {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   115
                // Only print non-default values so it easy to see
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   116
                // which options were added
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   117
                    StringJoiner options = new StringJoiner(", ");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   118
                    if (!toDisk) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   119
                        options.add("disk=false");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   120
                    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   121
                    if (maxAge != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   122
                        options.add("maxage=" + Utils.formatTimespan(maxAge, ""));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   123
                    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   124
                    if (maxSize != 0) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   125
                        options.add("maxsize=" + Utils.formatBytes(maxSize, ""));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   126
                    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   127
                    if (dumpOnExit) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   128
                        options.add("dumponexit=true");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   129
                    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   130
                    if (duration != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   131
                        options.add("duration=" + Utils.formatTimespan(duration, ""));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   132
                    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   133
                    if (destination != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   134
                        options.add("filename=" + destination.getText());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   135
                    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   136
                    String optionText = options.toString();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   137
                    if (optionText.length() != 0) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   138
                        optionText = "{" + optionText + "}";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   139
                    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   140
                    return "Started recording \"" + getName() + "\" (" + getId() + ") " + optionText;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   141
                });
50197
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   142
            newState = getState();
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   143
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   144
        notifyIfStateChanged(oldState, newState);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   145
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   146
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   147
    public boolean stop(String reason) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   148
        return stop(reason, null);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   149
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   150
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   151
    public boolean stop(String reason, WriteableUserPath alternativePath) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   152
        return stop(reason, alternativePath, Collections.emptyMap());
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
    boolean stop(String reason, WriteableUserPath alternativePath, Map<String, String> overlaySettings) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   156
        RecordingState oldState;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   157
        RecordingState newState;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   158
        synchronized (recorder) {
50197
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   159
            oldState = getState();
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   160
            if (stopTask != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   161
                stopTask.cancel();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   162
                stopTask = null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   163
            }
50194
2ae4cd9d809d 8198337: -XX:StartFlightRecording=dumponexit=true,disk=false doesn't work
egahlin
parents: 50113
diff changeset
   164
            recorder.stop(this, alternativePath);
50197
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   165
            String endText = reason == null ? "" : ". Reason \"" + reason + "\".";
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   166
            Logger.log(LogTag.JFR, LogLevel.INFO, "Stopped recording \"" + getName() + "\" (" + getId()+ ")" + endText);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   167
            this.stopTime = Instant.now();
50197
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   168
            newState = getState();
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   169
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   170
        WriteableUserPath dest = getDestination();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   171
        if (dest == null && alternativePath != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   172
            dest = alternativePath;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   173
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   174
        if (dest != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   175
            try {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   176
                copyTo(dest, reason, overlaySettings);
50197
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   177
                Logger.log(LogTag.JFR, LogLevel.INFO, "Wrote recording \"" + getName() + "\" (" + getId()+ ") to " + dest.getText());
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   178
                notifyIfStateChanged(newState, oldState);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   179
                close(); // remove if copied out
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   180
            } catch (IOException e) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   181
                // throw e; // BUG8925030
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   182
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   183
        } else {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   184
            notifyIfStateChanged(newState, oldState);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   185
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   186
        return true;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   187
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   188
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   189
    public void scheduleStart(Duration delay) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   190
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   191
            ensureOkForSchedule();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   192
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   193
            startTime = Instant.now().plus(delay);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   194
            LocalDateTime now = LocalDateTime.now().plus(delay);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   195
            setState(RecordingState.DELAYED);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   196
            startTask = createStartTask();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   197
            recorder.getTimer().schedule(startTask, delay.toMillis());
50197
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   198
            Logger.log(LogTag.JFR, LogLevel.INFO, "Scheduled recording \"" + getName() + "\" (" + getId()+ ") to start at " + now);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   199
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   200
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   201
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   202
    private void ensureOkForSchedule() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   203
        if (getState() != RecordingState.NEW) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   204
            throw new IllegalStateException("Only a new recoridng can be scheduled for start");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   205
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   206
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   207
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   208
    private TimerTask createStartTask() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   209
        // Taking ref. to recording here.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   210
        // Opens up for memory leaks.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   211
        return new TimerTask() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   212
            @Override
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   213
            public void run() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   214
                synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   215
                    if (getState() != RecordingState.DELAYED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   216
                        return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   217
                    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   218
                    start();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   219
                }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   220
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   221
        };
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   222
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   223
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   224
    void scheduleStart(Instant startTime) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   225
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   226
            ensureOkForSchedule();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   227
            this.startTime = startTime;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   228
            setState(RecordingState.DELAYED);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   229
            startTask = createStartTask();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   230
            recorder.getTimer().schedule(startTask, startTime.toEpochMilli());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   231
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   232
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   233
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   234
    public Map<String, String> getSettings() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   235
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   236
            return settings;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   237
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   238
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   239
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   240
    public long getSize() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   241
        return size;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   242
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   243
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   244
    public Instant getStopTime() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   245
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   246
            return stopTime;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   247
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   248
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   249
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   250
    public Instant getStartTime() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   251
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   252
            return startTime;
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
    public Long getMaxSize() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   257
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   258
            return maxSize;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   259
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   260
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   261
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   262
    public Duration getMaxAge() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   263
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   264
            return maxAge;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   265
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   266
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   267
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   268
    public String getName() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   269
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   270
            return name;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   271
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   272
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   273
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   274
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   275
    public RecordingState getState() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   276
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   277
            return state;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   278
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   279
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   280
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   281
    @Override
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   282
    public void close() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   283
        RecordingState oldState;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   284
        RecordingState newState;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   285
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   286
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   287
            oldState = getState();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   288
            if (RecordingState.CLOSED != getState()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   289
                if (startTask != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   290
                    startTask.cancel();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   291
                    startTask = null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   292
                }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   293
                recorder.finish(this);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   294
                for (RepositoryChunk c : chunks) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   295
                    removed(c);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   296
                }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   297
                chunks.clear();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   298
                setState(RecordingState.CLOSED);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   299
                Logger.log(LogTag.JFR, LogLevel.INFO, "Closed recording \"" + getName() + "\" (" + getId()+ ")");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   300
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   301
            newState = getState();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   302
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   303
        notifyIfStateChanged(newState, oldState);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   304
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   305
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   306
    public void copyTo(WriteableUserPath path, String reason, Map<String, String> dumpSettings) throws IOException {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   307
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   308
            RecordingState state = getState();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   309
            if (state == RecordingState.CLOSED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   310
                throw new IOException("Recording \"" + name + "\" (id=" + id + ") has been closed, no contents to write");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   311
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   312
            if (state == RecordingState.DELAYED || state == RecordingState.NEW) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   313
                throw new IOException("Recording \"" + name + "\" (id=" + id + ") has not started, no contents to write");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   314
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   315
            if (state == RecordingState.STOPPED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   316
                // have all we need, just write it
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   317
                dumpToFile(path, reason, getId());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   318
                return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   319
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   320
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   321
            // Recording is RUNNING, create a clone
50197
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   322
            try(PlatformRecording clone = recorder.newRecording(Collections.emptyMap(), 0)) {
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   323
                clone.setShouldWriteActiveRecordingEvent(false);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   324
                clone.setName(getName());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   325
                clone.setDestination(path);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   326
                clone.setToDisk(true);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   327
                // We purposely don't clone settings, since
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   328
                // a union a == a
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   329
                if (!isToDisk()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   330
                    // force memory contents to disk
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   331
                    clone.start();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   332
                } else {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   333
                    // using existing chunks on disk
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   334
                    for (RepositoryChunk c : chunks) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   335
                        clone.add(c);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   336
                    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   337
                    clone.setState(RecordingState.RUNNING);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   338
                    clone.setStartTime(getStartTime());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   339
                }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   340
                if (dumpSettings.isEmpty()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   341
                    clone.setSettings(getSettings());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   342
                    clone.stop(reason); // dumps to destination path here
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   343
                } else {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   344
                    // Risk of violating lock order here, since
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   345
                    // clone.stop() will take recorder lock inside
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   346
                    // metadata lock, but OK if we already
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   347
                    // have recorder lock when we entered metadata lock
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   348
                    Thread.holdsLock(recorder);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   349
                    synchronized(MetadataRepository.getInstance()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   350
                        Thread.holdsLock(recorder);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   351
                        Map<String, String> oldSettings = getSettings();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   352
                        Map<String, String> newSettings = new HashMap<>(oldSettings);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   353
                        // replace with dump settings
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   354
                        newSettings.putAll(dumpSettings);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   355
                        clone.setSettings(newSettings);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   356
                        clone.stop(reason);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   357
                    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   358
                }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   359
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   360
            return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   361
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   362
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   363
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   364
    private void dumpToFile(WriteableUserPath userPath, String reason, long id) throws IOException {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   365
        userPath.doPriviligedIO(() -> {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   366
            try (ChunksChannel cc = new ChunksChannel(chunks); FileChannel fc = FileChannel.open(userPath.getReal(), StandardOpenOption.WRITE, StandardOpenOption.APPEND)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   367
                cc.transferTo(fc);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   368
                fc.force(true);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   369
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   370
            return null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   371
        });
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   372
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   373
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   374
    public boolean isToDisk() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   375
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   376
            return toDisk;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   377
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   378
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   379
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   380
    public void setMaxSize(long maxSize) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   381
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   382
            if (getState() == RecordingState.CLOSED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   383
                throw new IllegalStateException("Can't set max age when recording is closed");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   384
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   385
            this.maxSize = maxSize;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   386
            trimToSize();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   387
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   388
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   389
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   390
   public void setDestination(WriteableUserPath userSuppliedPath) throws IOException {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   391
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   392
            if (Utils.isState(getState(), RecordingState.STOPPED, RecordingState.CLOSED)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   393
                throw new IllegalStateException("Destination can't be set on a recording that has been stopped/closed");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   394
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   395
            this.destination = userSuppliedPath;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   396
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   397
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   398
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   399
    public WriteableUserPath getDestination() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   400
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   401
            return destination;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   402
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   403
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   404
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   405
    void setState(RecordingState state) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   406
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   407
            this.state = state;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   408
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   409
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   410
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   411
    void setStartTime(Instant startTime) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   412
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   413
            this.startTime = startTime;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   414
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   415
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   416
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   417
    void setStopTime(Instant timeStamp) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   418
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   419
            stopTime = timeStamp;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   420
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   421
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   422
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   423
    public long getId() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   424
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   425
            return id;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   426
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   427
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   428
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   429
    public void setName(String name) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   430
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   431
            ensureNotClosed();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   432
            this.name = name;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   433
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   434
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   435
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   436
    private void ensureNotClosed() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   437
        if (getState() == RecordingState.CLOSED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   438
            throw new IllegalStateException("Can't change name on a closed recording");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   439
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   440
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   441
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   442
    public void setDumpOnExit(boolean dumpOnExit) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   443
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   444
            this.dumpOnExit = dumpOnExit;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   445
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   446
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   447
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   448
    public boolean getDumpOnExit() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   449
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   450
            return dumpOnExit;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   451
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   452
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   453
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   454
    public void setToDisk(boolean toDisk) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   455
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   456
            if (Utils.isState(getState(), RecordingState.NEW, RecordingState.DELAYED)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   457
                this.toDisk = toDisk;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   458
            } else {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   459
                throw new IllegalStateException("Recording option disk can't be changed after recording has started");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   460
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   461
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   462
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   463
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   464
    public void setSetting(String id, String value) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   465
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   466
            this.settings.put(id, value);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   467
            if (getState() == RecordingState.RUNNING) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   468
                recorder.updateSettings();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   469
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   470
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   471
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   472
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   473
    public void setSettings(Map<String, String> settings) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   474
        setSettings(settings, true);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   475
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   476
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   477
    private void setSettings(Map<String, String> settings, boolean update) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   478
        if (LogTag.JFR_SETTING.shouldLog(LogLevel.INFO.level) && update) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   479
            TreeMap<String, String> ordered = new TreeMap<>(settings);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   480
            Logger.log(LogTag.JFR_SETTING, LogLevel.INFO, "New settings for recording \"" + getName() + "\" (" + getId() + ")");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   481
            for (Map.Entry<String, String> entry : ordered.entrySet()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   482
                String text =  entry.getKey() + "=\"" + entry.getValue() + "\"";
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   483
                Logger.log(LogTag.JFR_SETTING, LogLevel.INFO, text);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   484
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   485
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   486
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   487
            this.settings = new LinkedHashMap<>(settings);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   488
            if (getState() == RecordingState.RUNNING && update) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   489
                recorder.updateSettings();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   490
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   491
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   492
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   493
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   494
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   495
    private void notifyIfStateChanged(RecordingState newState, RecordingState oldState) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   496
        if (oldState == newState) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   497
            return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   498
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   499
        for (FlightRecorderListener cl : PlatformRecorder.getListeners()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   500
            try {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   501
                cl.recordingStateChanged(getRecording());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   502
            } catch (RuntimeException re) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   503
                Logger.log(JFR, WARN, "Error notifying recorder listener:" + re.getMessage());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   504
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   505
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   506
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   507
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   508
    public void setRecording(Recording recording) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   509
        this.recording = recording;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   510
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   511
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   512
    public Recording getRecording() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   513
        return recording;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   514
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   515
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   516
    @Override
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   517
    public String toString() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   518
        return getName() + " (id=" + getId() + ") " + getState();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   519
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   520
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   521
    public void setConfiguration(Configuration c) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   522
        setSettings(c.getSettings());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   523
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   524
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   525
    public void setMaxAge(Duration maxAge) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   526
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   527
            if (getState() == RecordingState.CLOSED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   528
                throw new IllegalStateException("Can't set max age when recording is closed");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   529
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   530
            this.maxAge = maxAge;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   531
            if (maxAge != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   532
                trimToAge(Instant.now().minus(maxAge));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   533
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   534
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   535
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   536
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   537
    void appendChunk(RepositoryChunk chunk) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   538
        if (!chunk.isFinished()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   539
            throw new Error("not finished chunk " + chunk.getStartTime());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   540
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   541
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   542
            if (!toDisk) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   543
                return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   544
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   545
            if (maxAge != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   546
                trimToAge(chunk.getEndTime().minus(maxAge));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   547
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   548
            chunks.addLast(chunk);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   549
            added(chunk);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   550
            trimToSize();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   551
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   552
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   553
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   554
    private void trimToSize() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   555
        if (maxSize == 0) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   556
            return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   557
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   558
        while (size > maxSize && chunks.size() > 1) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   559
            RepositoryChunk c = chunks.removeFirst();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   560
            removed(c);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   561
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   562
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   563
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   564
    private void trimToAge(Instant oldest) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   565
        while (!chunks.isEmpty()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   566
            RepositoryChunk oldestChunk = chunks.peek();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   567
            if (oldestChunk.getEndTime().isAfter(oldest)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   568
                return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   569
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   570
            chunks.removeFirst();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   571
            removed(oldestChunk);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   572
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   573
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   574
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   575
    void add(RepositoryChunk c) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   576
        chunks.add(c);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   577
        added(c);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   578
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   579
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   580
    private void added(RepositoryChunk c) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   581
        c.use();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   582
        size += c.getSize();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   583
        Logger.log(JFR, DEBUG,  ()-> "Recording \"" + name + "\" (" + id + ") added chunk " + c.toString() + ", current size=" + size);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   584
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   585
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   586
    private void removed(RepositoryChunk c) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   587
        size -= c.getSize();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   588
        Logger.log(JFR, DEBUG,  ()->  "Recording \"" + name + "\" (" + id + ") removed chunk " + c.toString() + ", current size=" + size);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   589
        c.release();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   590
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   591
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   592
    public List<RepositoryChunk> getChunks() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   593
        return chunks;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   594
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   595
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   596
    public InputStream open(Instant start, Instant end) throws IOException {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   597
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   598
            if (getState() != RecordingState.STOPPED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   599
                throw new IOException("Recording must be stopped before it can be read.");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   600
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   601
            List<RepositoryChunk> chunksToUse = new ArrayList<RepositoryChunk>();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   602
            for (RepositoryChunk chunk : chunks) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   603
                if (chunk.isFinished()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   604
                    Instant chunkStart = chunk.getStartTime();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   605
                    Instant chunkEnd = chunk.getEndTime();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   606
                    if (start == null || !chunkEnd.isBefore(start)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   607
                        if (end == null || !chunkStart.isAfter(end)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   608
                            chunksToUse.add(chunk);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   609
                        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   610
                    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   611
                }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   612
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   613
            if (chunksToUse.isEmpty()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   614
                return null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   615
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   616
            return new ChunkInputStream(chunksToUse);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   617
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   618
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   619
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   620
    public Duration getDuration() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   621
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   622
            return duration;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   623
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   624
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   625
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   626
    void setInternalDuration(Duration duration) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   627
        this.duration = duration;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   628
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   629
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   630
    public void setDuration(Duration duration) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   631
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   632
            if (Utils.isState(getState(), RecordingState.STOPPED, RecordingState.CLOSED)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   633
                throw new IllegalStateException("Duration can't be set after a recording has been stopped/closed");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   634
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   635
            setInternalDuration(duration);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   636
            if (getState() != RecordingState.NEW) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   637
                updateTimer();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   638
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   639
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   640
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   641
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   642
    void updateTimer() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   643
        if (stopTask != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   644
            stopTask.cancel();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   645
            stopTask = null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   646
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   647
        if (getState() == RecordingState.CLOSED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   648
            return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   649
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   650
        if (duration != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   651
            stopTask = createStopTask();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   652
            recorder.getTimer().schedule(stopTask, new Date(startTime.plus(duration).toEpochMilli()));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   653
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   654
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   655
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   656
    TimerTask createStopTask() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   657
        return new TimerTask() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   658
            @Override
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   659
            public void run() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   660
                try {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   661
                    stop("End of duration reached");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   662
                } catch (Throwable t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   663
                    // Prevent malicious user to propagate exception callback in the wrong context
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   664
                    Logger.log(LogTag.JFR, LogLevel.ERROR, "Could not stop recording. " + t.getMessage());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   665
                }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   666
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   667
        };
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   668
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   669
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   670
    public Recording newCopy(boolean stop) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   671
        return recorder.newCopy(this, stop);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   672
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   673
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   674
    void setStopTask(TimerTask stopTask) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   675
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   676
            this.stopTask = stopTask;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   677
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   678
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   679
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   680
    void clearDestination() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   681
       destination = null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   682
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   683
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   684
    public AccessControlContext getNoDestinationDumpOnExitAccessControlContext() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   685
        return noDestinationDumpOnExitAccessControlContext;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   686
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   687
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   688
    void setShouldWriteActiveRecordingEvent(boolean shouldWrite) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   689
       this.shuoldWriteActiveRecordingEvent = shouldWrite;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   690
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   691
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   692
    boolean shouldWriteMetadataEvent() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   693
        return shuoldWriteActiveRecordingEvent;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   694
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   695
}