src/jdk.jfr/share/classes/jdk/jfr/internal/PlatformRecording.java
author mgronlun
Wed, 30 Oct 2019 19:43:52 +0100
changeset 58863 c16ac7a2eba4
parent 58480 8ca46e186a63
child 59010 d4cae08239f7
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: 58480
diff changeset
     2
 * Copyright (c) 2016, 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
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.LinkedHashMap;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    45
import java.util.LinkedList;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    46
import java.util.List;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    47
import java.util.Map;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    48
import java.util.StringJoiner;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    49
import java.util.TimerTask;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    50
import java.util.TreeMap;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    51
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    52
import jdk.jfr.Configuration;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    53
import jdk.jfr.FlightRecorderListener;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    54
import jdk.jfr.Recording;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    55
import jdk.jfr.RecordingState;
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
    56
import jdk.jfr.internal.SecuritySupport.SafePath;
50113
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;
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
    73
    private SafePath dumpOnExitDirectory = new SafePath(".");
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    74
    // Timestamp information
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    75
    private Instant stopTime;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    76
    private Instant startTime;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    77
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    78
    // Misc, information
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    79
    private RecordingState state = RecordingState.NEW;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    80
    private long size;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    81
    private final LinkedList<RepositoryChunk> chunks = new LinkedList<>();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    82
    private volatile Recording recording;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    83
    private TimerTask stopTask;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    84
    private TimerTask startTask;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    85
    private AccessControlContext noDestinationDumpOnExitAccessControlContext;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    86
    private boolean shuoldWriteActiveRecordingEvent = true;
58863
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
    87
    private Duration flushInterval = Duration.ofSeconds(1);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    88
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    89
    PlatformRecording(PlatformRecorder recorder, long id) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    90
        // Typically the access control context is taken
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    91
        // when you call dump(Path) or setDdestination(Path),
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    92
        // but if no destination is set and dumponexit=true
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    93
        // the control context of the recording is taken when the
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
    94
        // Recording object is constructed. This works well for
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    95
        // -XX:StartFlightRecording and JFR.dump
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    96
        this.noDestinationDumpOnExitAccessControlContext = AccessController.getContext();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    97
        this.id = id;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    98
        this.recorder = recorder;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
    99
        this.name = String.valueOf(id);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   100
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   101
58863
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   102
    public long start() {
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   103
        RecordingState oldState;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   104
        RecordingState newState;
58863
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   105
        long startNanos = -1;
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   106
        synchronized (recorder) {
50197
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   107
            oldState = getState();
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   108
            if (!Utils.isBefore(state, RecordingState.RUNNING)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   109
                throw new IllegalStateException("Recording can only be started once.");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   110
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   111
            if (startTask != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   112
                startTask.cancel();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   113
                startTask = null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   114
                startTime = null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   115
            }
58863
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   116
            startNanos = recorder.start(this);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   117
            Logger.log(LogTag.JFR, LogLevel.INFO, () -> {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   118
                // Only print non-default values so it easy to see
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   119
                // which options were added
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   120
                StringJoiner options = new StringJoiner(", ");
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   121
                if (!toDisk) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   122
                    options.add("disk=false");
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   123
                }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   124
                if (maxAge != null) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   125
                    options.add("maxage=" + Utils.formatTimespan(maxAge, ""));
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   126
                }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   127
                if (maxSize != 0) {
53013
c8b2a408628b 8215175: Inconsistencies in JFR event metadata
egahlin
parents: 52413
diff changeset
   128
                    options.add("maxsize=" + Utils.formatBytesCompact(maxSize));
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   129
                }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   130
                if (dumpOnExit) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   131
                    options.add("dumponexit=true");
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   132
                }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   133
                if (duration != null) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   134
                    options.add("duration=" + Utils.formatTimespan(duration, ""));
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   135
                }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   136
                if (destination != null) {
55256
3b22c7e00573 8224217: RecordingInfo should use textual representation of path
egahlin
parents: 53013
diff changeset
   137
                    options.add("filename=" + destination.getRealPathText());
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   138
                }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   139
                String optionText = options.toString();
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   140
                if (optionText.length() != 0) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   141
                    optionText = "{" + optionText + "}";
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   142
                }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   143
                return "Started recording \"" + getName() + "\" (" + getId() + ") " + optionText;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   144
            });
50197
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   145
            newState = getState();
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   146
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   147
        notifyIfStateChanged(oldState, newState);
58863
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   148
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   149
        return startNanos;
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   150
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   151
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   152
    public boolean stop(String reason) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   153
        RecordingState oldState;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   154
        RecordingState newState;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   155
        synchronized (recorder) {
50197
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   156
            oldState = getState();
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   157
            if (stopTask != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   158
                stopTask.cancel();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   159
                stopTask = null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   160
            }
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   161
            recorder.stop(this);
50197
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   162
            String endText = reason == null ? "" : ". Reason \"" + reason + "\".";
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   163
            Logger.log(LogTag.JFR, LogLevel.INFO, "Stopped recording \"" + getName() + "\" (" + getId() + ")" + endText);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   164
            this.stopTime = Instant.now();
50197
f4735ff8d17d 8195818: JFR.start should increase autogenerated name by one
egahlin
parents: 50194
diff changeset
   165
            newState = getState();
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   166
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   167
        WriteableUserPath dest = getDestination();
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   168
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   169
        if (dest != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   170
            try {
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   171
                dumpStopped(dest);
55256
3b22c7e00573 8224217: RecordingInfo should use textual representation of path
egahlin
parents: 53013
diff changeset
   172
                Logger.log(LogTag.JFR, LogLevel.INFO, "Wrote recording \"" + getName() + "\" (" + getId() + ") to " + dest.getRealPathText());
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   173
                notifyIfStateChanged(newState, oldState);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   174
                close(); // remove if copied out
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   175
            } catch(IOException e) {
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   176
                // throw e; // BUG8925030
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   177
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   178
        } else {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   179
            notifyIfStateChanged(newState, oldState);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   180
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   181
        return true;
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
    public void scheduleStart(Duration delay) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   185
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   186
            ensureOkForSchedule();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   187
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   188
            startTime = Instant.now().plus(delay);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   189
            LocalDateTime now = LocalDateTime.now().plus(delay);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   190
            setState(RecordingState.DELAYED);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   191
            startTask = createStartTask();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   192
            recorder.getTimer().schedule(startTask, delay.toMillis());
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   193
            Logger.log(LogTag.JFR, LogLevel.INFO, "Scheduled recording \"" + getName() + "\" (" + getId() + ") to start at " + now);
50113
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
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   197
    private void ensureOkForSchedule() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   198
        if (getState() != RecordingState.NEW) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   199
            throw new IllegalStateException("Only a new recoridng can be scheduled for start");
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
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   203
    private TimerTask createStartTask() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   204
        // Taking ref. to recording here.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   205
        // Opens up for memory leaks.
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   206
        return new TimerTask() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   207
            @Override
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   208
            public void run() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   209
                synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   210
                    if (getState() != RecordingState.DELAYED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   211
                        return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   212
                    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   213
                    start();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   214
                }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   215
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   216
        };
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   217
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   218
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   219
    void scheduleStart(Instant startTime) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   220
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   221
            ensureOkForSchedule();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   222
            this.startTime = startTime;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   223
            setState(RecordingState.DELAYED);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   224
            startTask = createStartTask();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   225
            recorder.getTimer().schedule(startTask, startTime.toEpochMilli());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   226
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   227
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   228
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   229
    public Map<String, String> getSettings() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   230
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   231
            return settings;
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
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   235
    public long getSize() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   236
        return size;
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
    public Instant getStopTime() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   240
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   241
            return stopTime;
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
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   245
    public Instant getStartTime() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   246
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   247
            return startTime;
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
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   251
    public Long getMaxSize() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   252
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   253
            return maxSize;
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
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   257
    public Duration getMaxAge() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   258
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   259
            return maxAge;
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
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   263
    public String getName() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   264
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   265
            return name;
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
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   269
    public RecordingState getState() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   270
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   271
            return state;
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
    @Override
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   276
    public void close() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   277
        RecordingState oldState;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   278
        RecordingState newState;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   279
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   280
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   281
            oldState = getState();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   282
            if (RecordingState.CLOSED != getState()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   283
                if (startTask != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   284
                    startTask.cancel();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   285
                    startTask = null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   286
                }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   287
                recorder.finish(this);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   288
                for (RepositoryChunk c : chunks) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   289
                    removed(c);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   290
                }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   291
                chunks.clear();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   292
                setState(RecordingState.CLOSED);
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   293
                Logger.log(LogTag.JFR, LogLevel.INFO, "Closed recording \"" + getName() + "\" (" + getId() + ")");
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   294
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   295
            newState = getState();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   296
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   297
        notifyIfStateChanged(newState, oldState);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   298
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   299
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   300
    // To be used internally when doing dumps.
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   301
    // Caller must have recorder lock and close recording before releasing lock
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   302
    public PlatformRecording newSnapshotClone(String reason, Boolean pathToGcRoots) throws IOException {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   303
        if(!Thread.holdsLock(recorder)) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   304
            throw new InternalError("Caller must have recorder lock");
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   305
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   306
        RecordingState state = getState();
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   307
        if (state == RecordingState.CLOSED) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   308
            throw new IOException("Recording \"" + name + "\" (id=" + id + ") has been closed, no contents to write");
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   309
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   310
        if (state == RecordingState.DELAYED || state == RecordingState.NEW) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   311
            throw new IOException("Recording \"" + name + "\" (id=" + id + ") has not started, no contents to write");
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   312
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   313
        if (state == RecordingState.STOPPED) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   314
            PlatformRecording clone = recorder.newTemporaryRecording();
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   315
            for (RepositoryChunk r : chunks) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   316
                clone.add(r);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   317
            }
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   318
            return clone;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   319
        }
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   320
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   321
        // Recording is RUNNING, create a clone
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   322
        PlatformRecording clone = recorder.newTemporaryRecording();
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   323
        clone.setShouldWriteActiveRecordingEvent(false);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   324
        clone.setName(getName());
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   325
        clone.setToDisk(true);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   326
        // We purposely don't clone settings here, since
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   327
        // a union a == a
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   328
        if (!isToDisk()) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   329
            // force memory contents to disk
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   330
            clone.start();
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   331
        } else {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   332
            // using existing chunks on disk
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   333
            for (RepositoryChunk c : chunks) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   334
                clone.add(c);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   335
            }
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   336
            clone.setState(RecordingState.RUNNING);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   337
            clone.setStartTime(getStartTime());
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   338
        }
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   339
        if (pathToGcRoots == null) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   340
            clone.setSettings(getSettings()); // needed for old object sample
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   341
            clone.stop(reason); // dumps to destination path here
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   342
        } else {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   343
            // Risk of violating lock order here, since
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   344
            // clone.stop() will take recorder lock inside
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   345
            // metadata lock, but OK if we already
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   346
            // have recorder lock when we entered metadata lock
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   347
            synchronized (MetadataRepository.getInstance()) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   348
                clone.setSettings(OldObjectSample.createSettingsForSnapshot(this, pathToGcRoots));
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   349
                clone.stop(reason);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   350
            }
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   351
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   352
        return clone;
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   353
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   354
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   355
    public boolean isToDisk() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   356
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   357
            return toDisk;
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
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   361
    public void setMaxSize(long maxSize) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   362
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   363
            if (getState() == RecordingState.CLOSED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   364
                throw new IllegalStateException("Can't set max age when recording is closed");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   365
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   366
            this.maxSize = maxSize;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   367
            trimToSize();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   368
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   369
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   370
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   371
    public void setDestination(WriteableUserPath userSuppliedPath) throws IOException {
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   372
        synchronized (recorder) {
58480
8ca46e186a63 8225694: Destination option missing in FlightRecorderMXBeanImpl
cito
parents: 55378
diff changeset
   373
            checkSetDestination(userSuppliedPath);
8ca46e186a63 8225694: Destination option missing in FlightRecorderMXBeanImpl
cito
parents: 55378
diff changeset
   374
            this.destination = userSuppliedPath;
8ca46e186a63 8225694: Destination option missing in FlightRecorderMXBeanImpl
cito
parents: 55378
diff changeset
   375
        }
8ca46e186a63 8225694: Destination option missing in FlightRecorderMXBeanImpl
cito
parents: 55378
diff changeset
   376
    }
8ca46e186a63 8225694: Destination option missing in FlightRecorderMXBeanImpl
cito
parents: 55378
diff changeset
   377
8ca46e186a63 8225694: Destination option missing in FlightRecorderMXBeanImpl
cito
parents: 55378
diff changeset
   378
    public void checkSetDestination(WriteableUserPath userSuppliedPath) throws IOException {
8ca46e186a63 8225694: Destination option missing in FlightRecorderMXBeanImpl
cito
parents: 55378
diff changeset
   379
        synchronized (recorder) {
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   380
            if (Utils.isState(getState(), RecordingState.STOPPED, RecordingState.CLOSED)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   381
                throw new IllegalStateException("Destination can't be set on a recording that has been stopped/closed");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   382
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   383
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   384
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   385
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   386
    public WriteableUserPath getDestination() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   387
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   388
            return destination;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   389
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   390
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   391
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   392
    void setState(RecordingState state) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   393
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   394
            this.state = state;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   395
        }
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
    void setStartTime(Instant startTime) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   399
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   400
            this.startTime = startTime;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   401
        }
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
    void setStopTime(Instant timeStamp) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   405
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   406
            stopTime = timeStamp;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   407
        }
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
    public long getId() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   411
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   412
            return id;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   413
        }
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
    public void setName(String name) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   417
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   418
            ensureNotClosed();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   419
            this.name = name;
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
    private void ensureNotClosed() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   424
        if (getState() == RecordingState.CLOSED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   425
            throw new IllegalStateException("Can't change name on a closed recording");
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 setDumpOnExit(boolean dumpOnExit) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   430
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   431
            this.dumpOnExit = dumpOnExit;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   432
        }
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
    public boolean getDumpOnExit() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   436
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   437
            return dumpOnExit;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   438
        }
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
    public void setToDisk(boolean toDisk) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   442
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   443
            if (Utils.isState(getState(), RecordingState.NEW, RecordingState.DELAYED)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   444
                this.toDisk = toDisk;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   445
            } else {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   446
                throw new IllegalStateException("Recording option disk can't be changed after recording has started");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   447
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   448
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   449
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   450
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   451
    public void setSetting(String id, String value) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   452
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   453
            this.settings.put(id, value);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   454
            if (getState() == RecordingState.RUNNING) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   455
                recorder.updateSettings();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   456
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   457
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   458
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   459
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   460
    public void setSettings(Map<String, String> settings) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   461
        setSettings(settings, true);
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
    private void setSettings(Map<String, String> settings, boolean update) {
52413
6372f5af9612 8209960: -Xlog:jfr* doesn't work with the JFR
egahlin
parents: 50745
diff changeset
   465
        if (Logger.shouldLog(LogTag.JFR_SETTING, LogLevel.INFO) && update) {
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   466
            TreeMap<String, String> ordered = new TreeMap<>(settings);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   467
            Logger.log(LogTag.JFR_SETTING, LogLevel.INFO, "New settings for recording \"" + getName() + "\" (" + getId() + ")");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   468
            for (Map.Entry<String, String> entry : ordered.entrySet()) {
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   469
                String text = entry.getKey() + "=\"" + entry.getValue() + "\"";
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   470
                Logger.log(LogTag.JFR_SETTING, LogLevel.INFO, text);
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
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   474
            this.settings = new LinkedHashMap<>(settings);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   475
            if (getState() == RecordingState.RUNNING && update) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   476
                recorder.updateSettings();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   477
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   478
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   479
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   480
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   481
    private void notifyIfStateChanged(RecordingState newState, RecordingState oldState) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   482
        if (oldState == newState) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   483
            return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   484
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   485
        for (FlightRecorderListener cl : PlatformRecorder.getListeners()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   486
            try {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   487
                cl.recordingStateChanged(getRecording());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   488
            } catch (RuntimeException re) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   489
                Logger.log(JFR, WARN, "Error notifying recorder listener:" + re.getMessage());
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
    public void setRecording(Recording recording) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   495
        this.recording = recording;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   496
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   497
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   498
    public Recording getRecording() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   499
        return recording;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   500
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   501
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   502
    @Override
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   503
    public String toString() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   504
        return getName() + " (id=" + getId() + ") " + getState();
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
    public void setConfiguration(Configuration c) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   508
        setSettings(c.getSettings());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   509
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   510
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   511
    public void setMaxAge(Duration maxAge) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   512
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   513
            if (getState() == RecordingState.CLOSED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   514
                throw new IllegalStateException("Can't set max age when recording is closed");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   515
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   516
            this.maxAge = maxAge;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   517
            if (maxAge != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   518
                trimToAge(Instant.now().minus(maxAge));
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
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   522
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   523
    void appendChunk(RepositoryChunk chunk) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   524
        if (!chunk.isFinished()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   525
            throw new Error("not finished chunk " + chunk.getStartTime());
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   526
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   527
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   528
            if (!toDisk) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   529
                return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   530
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   531
            if (maxAge != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   532
                trimToAge(chunk.getEndTime().minus(maxAge));
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   533
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   534
            chunks.addLast(chunk);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   535
            added(chunk);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   536
            trimToSize();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   537
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   538
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   539
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   540
    private void trimToSize() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   541
        if (maxSize == 0) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   542
            return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   543
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   544
        while (size > maxSize && chunks.size() > 1) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   545
            RepositoryChunk c = chunks.removeFirst();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   546
            removed(c);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   547
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   548
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   549
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   550
    private void trimToAge(Instant oldest) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   551
        while (!chunks.isEmpty()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   552
            RepositoryChunk oldestChunk = chunks.peek();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   553
            if (oldestChunk.getEndTime().isAfter(oldest)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   554
                return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   555
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   556
            chunks.removeFirst();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   557
            removed(oldestChunk);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   558
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   559
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   560
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   561
    void add(RepositoryChunk c) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   562
        chunks.add(c);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   563
        added(c);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   564
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   565
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   566
    private void added(RepositoryChunk c) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   567
        c.use();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   568
        size += c.getSize();
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   569
        Logger.log(JFR, DEBUG, () -> "Recording \"" + name + "\" (" + id + ") added chunk " + c.toString() + ", current size=" + size);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   570
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   571
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   572
    private void removed(RepositoryChunk c) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   573
        size -= c.getSize();
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   574
        Logger.log(JFR, DEBUG, () -> "Recording \"" + name + "\" (" + id + ") removed chunk " + c.toString() + ", current size=" + size);
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   575
        c.release();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   576
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   577
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   578
    public List<RepositoryChunk> getChunks() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   579
        return chunks;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   580
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   581
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   582
    public InputStream open(Instant start, Instant end) throws IOException {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   583
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   584
            if (getState() != RecordingState.STOPPED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   585
                throw new IOException("Recording must be stopped before it can be read.");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   586
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   587
            List<RepositoryChunk> chunksToUse = new ArrayList<RepositoryChunk>();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   588
            for (RepositoryChunk chunk : chunks) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   589
                if (chunk.isFinished()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   590
                    Instant chunkStart = chunk.getStartTime();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   591
                    Instant chunkEnd = chunk.getEndTime();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   592
                    if (start == null || !chunkEnd.isBefore(start)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   593
                        if (end == null || !chunkStart.isAfter(end)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   594
                            chunksToUse.add(chunk);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   595
                        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   596
                    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   597
                }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   598
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   599
            if (chunksToUse.isEmpty()) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   600
                return null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   601
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   602
            return new ChunkInputStream(chunksToUse);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   603
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   604
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   605
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   606
    public Duration getDuration() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   607
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   608
            return duration;
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
    void setInternalDuration(Duration duration) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   613
        this.duration = duration;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   614
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   615
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   616
    public void setDuration(Duration duration) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   617
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   618
            if (Utils.isState(getState(), RecordingState.STOPPED, RecordingState.CLOSED)) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   619
                throw new IllegalStateException("Duration can't be set after a recording has been stopped/closed");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   620
            }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   621
            setInternalDuration(duration);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   622
            if (getState() != RecordingState.NEW) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   623
                updateTimer();
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
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   627
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   628
    void updateTimer() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   629
        if (stopTask != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   630
            stopTask.cancel();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   631
            stopTask = null;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   632
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   633
        if (getState() == RecordingState.CLOSED) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   634
            return;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   635
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   636
        if (duration != null) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   637
            stopTask = createStopTask();
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   638
            recorder.getTimer().schedule(stopTask, new Date(startTime.plus(duration).toEpochMilli()));
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
    TimerTask createStopTask() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   643
        return new TimerTask() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   644
            @Override
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   645
            public void run() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   646
                try {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   647
                    stop("End of duration reached");
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   648
                } catch (Throwable t) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   649
                    // Prevent malicious user to propagate exception callback in the wrong context
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   650
                    Logger.log(LogTag.JFR, LogLevel.ERROR, "Could not stop recording.");
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   651
                }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   652
            }
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
    public Recording newCopy(boolean stop) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   657
        return recorder.newCopy(this, stop);
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   658
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   659
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   660
    void setStopTask(TimerTask stopTask) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   661
        synchronized (recorder) {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   662
            this.stopTask = stopTask;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   663
        }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   664
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   665
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   666
    void clearDestination() {
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   667
        destination = null;
50113
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 AccessControlContext getNoDestinationDumpOnExitAccessControlContext() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   671
        return noDestinationDumpOnExitAccessControlContext;
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 setShouldWriteActiveRecordingEvent(boolean shouldWrite) {
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   675
        this.shuoldWriteActiveRecordingEvent = shouldWrite;
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   676
    }
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   677
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   678
    boolean shouldWriteMetadataEvent() {
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   679
        return shuoldWriteActiveRecordingEvent;
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   680
    }
50745
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   681
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   682
    // Dump running and stopped recordings
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   683
    public void dump(WriteableUserPath writeableUserPath) throws IOException {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   684
        synchronized (recorder) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   685
            try(PlatformRecording p = newSnapshotClone("Dumped by user", null))  {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   686
                p.dumpStopped(writeableUserPath);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   687
            }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   688
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   689
    }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   690
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   691
    public void dumpStopped(WriteableUserPath userPath) throws IOException {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   692
        synchronized (recorder) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   693
                userPath.doPriviligedIO(() -> {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   694
                    try (ChunksChannel cc = new ChunksChannel(chunks); FileChannel fc = FileChannel.open(userPath.getReal(), StandardOpenOption.WRITE, StandardOpenOption.APPEND)) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   695
                        cc.transferTo(fc);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   696
                        fc.force(true);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   697
                    }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   698
                    return null;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   699
                });
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   700
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   701
    }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   702
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   703
    public void filter(Instant begin, Instant end, Long maxSize) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   704
        synchronized (recorder) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   705
            List<RepositoryChunk> result = removeAfter(end, removeBefore(begin, new ArrayList<>(chunks)));
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   706
            if (maxSize != null) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   707
                if (begin != null && end == null) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   708
                    result = reduceFromBeginning(maxSize, result);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   709
                } else {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   710
                    result = reduceFromEnd(maxSize, result);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   711
                }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   712
            }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   713
            int size = 0;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   714
            for (RepositoryChunk r : result) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   715
                size += r.getSize();
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   716
                r.use();
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   717
            }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   718
            this.size = size;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   719
            for (RepositoryChunk r : chunks) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   720
                r.release();
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   721
            }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   722
            chunks.clear();
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   723
            chunks.addAll(result);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   724
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   725
    }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   726
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   727
    private static List<RepositoryChunk> removeBefore(Instant time, List<RepositoryChunk> input) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   728
        if (time == null) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   729
            return input;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   730
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   731
        List<RepositoryChunk> result = new ArrayList<>(input.size());
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   732
        for (RepositoryChunk r : input) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   733
            if (!r.getEndTime().isBefore(time)) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   734
                result.add(r);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   735
            }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   736
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   737
        return result;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   738
    }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   739
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   740
    private static List<RepositoryChunk> removeAfter(Instant time, List<RepositoryChunk> input) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   741
        if (time == null) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   742
            return input;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   743
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   744
        List<RepositoryChunk> result = new ArrayList<>(input.size());
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   745
        for (RepositoryChunk r : input) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   746
            if (!r.getStartTime().isAfter(time)) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   747
                result.add(r);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   748
            }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   749
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   750
        return result;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   751
    }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   752
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   753
    private static List<RepositoryChunk> reduceFromBeginning(Long maxSize, List<RepositoryChunk> input) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   754
        if (maxSize == null || input.isEmpty()) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   755
            return input;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   756
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   757
        List<RepositoryChunk> result = new ArrayList<>(input.size());
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   758
        long total = 0;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   759
        for (RepositoryChunk r : input) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   760
            total += r.getSize();
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   761
            if (total > maxSize) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   762
                break;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   763
            }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   764
            result.add(r);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   765
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   766
        // always keep at least one chunk
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   767
        if (result.isEmpty()) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   768
            result.add(input.get(0));
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   769
        }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   770
        return result;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   771
    }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   772
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   773
    private static List<RepositoryChunk> reduceFromEnd(Long maxSize, List<RepositoryChunk> input) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   774
        Collections.reverse(input);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   775
        List<RepositoryChunk> result = reduceFromBeginning(maxSize, input);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   776
        Collections.reverse(result);
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   777
        return result;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   778
    }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   779
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   780
    public void setDumpOnExitDirectory(SafePath directory) {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   781
       this.dumpOnExitDirectory = directory;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   782
    }
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   783
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   784
    public SafePath getDumpOnExitDirectory()  {
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   785
        return this.dumpOnExitDirectory;
a390cbb82d47 8203929: Limit amount of data for JFR.dump
egahlin
parents: 50197
diff changeset
   786
    }
58863
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   787
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   788
    public void setFlushInterval(Duration interval) {
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   789
        synchronized (recorder) {
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   790
            if (getState() == RecordingState.CLOSED) {
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   791
                throw new IllegalStateException("Can't set stream interval when recording is closed");
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   792
            }
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   793
            this.flushInterval = interval;
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   794
        }
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   795
    }
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   796
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   797
    public Duration getFlushInterval() {
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   798
        synchronized (recorder) {
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   799
            return flushInterval;
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   800
        }
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   801
    }
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   802
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   803
    public long getStreamIntervalMillis() {
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   804
        synchronized (recorder) {
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   805
            if (flushInterval != null) {
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   806
                return flushInterval.toMillis();
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   807
            }
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   808
            return Long.MAX_VALUE;
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   809
        }
c16ac7a2eba4 8226511: Implement JFR Event Streaming
mgronlun
parents: 58480
diff changeset
   810
    }
50113
caf115bb98ad 8199712: Flight Recorder
egahlin
parents:
diff changeset
   811
}