From bbc2e9510bb32d69d823bd572b6c0c02bd2335af Mon Sep 17 00:00:00 2001 From: Erik Gahlin Date: Tue, 5 Jan 2021 13:39:57 +0000 Subject: [PATCH] 8257906: JFR: RecordingStream leaks memory Reviewed-by: mgronlun Backport-of: 3c6648501589bf36945340cb1e82c833ebd7485d --- .../jdk/jfr/consumer/RecordingStream.java | 43 +++++++++++++++++-- .../jfr/internal/consumer/ChunkParser.java | 12 ++++++ .../consumer/EventDirectoryStream.java | 1 + .../internal/consumer/EventFileStream.java | 1 + 4 files changed, 54 insertions(+), 3 deletions(-) diff --git a/src/jdk.jfr/share/classes/jdk/jfr/consumer/RecordingStream.java b/src/jdk.jfr/share/classes/jdk/jfr/consumer/RecordingStream.java index 78fe38d6d34..7938fc22831 100644 --- a/src/jdk.jfr/share/classes/jdk/jfr/consumer/RecordingStream.java +++ b/src/jdk.jfr/share/classes/jdk/jfr/consumer/RecordingStream.java @@ -45,7 +45,6 @@ import jdk.jfr.internal.PrivateAccess; import jdk.jfr.internal.SecuritySupport; import jdk.jfr.internal.Utils; import jdk.jfr.internal.consumer.EventDirectoryStream; -import jdk.jfr.internal.consumer.JdkJfrConsumer; /** * A recording stream produces events from the current JVM (Java Virtual @@ -68,9 +67,27 @@ import jdk.jfr.internal.consumer.JdkJfrConsumer; */ public final class RecordingStream implements AutoCloseable, EventStream { + final static class ChunkConsumer implements Consumer { + + private final Recording recording; + + ChunkConsumer(Recording recording) { + this.recording = recording; + } + + @Override + public void accept(Long endNanos) { + Instant t = Utils.epochNanosToInstant(endNanos); + PlatformRecording p = PrivateAccess.getInstance().getPlatformRecording(recording); + p.removeBefore(t); + } + } + private final Recording recording; private final Instant creationTime; private final EventDirectoryStream directoryStream; + private long maxSize; + private Duration maxAge; /** * Creates an event stream for the current JVM (Java Virtual Machine). @@ -247,7 +264,11 @@ public final class RecordingStream implements AutoCloseable, EventStream { * state */ public void setMaxAge(Duration maxAge) { - recording.setMaxAge(maxAge); + synchronized (directoryStream) { + recording.setMaxAge(maxAge); + this.maxAge = maxAge; + updateOnCompleteHandler(); + } } /** @@ -270,7 +291,11 @@ public final class RecordingStream implements AutoCloseable, EventStream { * @throws IllegalStateException if the recording is in {@code CLOSED} state */ public void setMaxSize(long maxSize) { - recording.setMaxSize(maxSize); + synchronized (directoryStream) { + recording.setMaxSize(maxSize); + this.maxSize = maxSize; + updateOnCompleteHandler(); + } } @Override @@ -320,6 +345,7 @@ public final class RecordingStream implements AutoCloseable, EventStream { @Override public void close() { + directoryStream.setChunkCompleteHandler(null); recording.close(); directoryStream.close(); } @@ -333,6 +359,7 @@ public final class RecordingStream implements AutoCloseable, EventStream { public void start() { PlatformRecording pr = PrivateAccess.getInstance().getPlatformRecording(recording); long startNanos = pr.start(); + updateOnCompleteHandler(); directoryStream.start(startNanos); } @@ -363,6 +390,7 @@ public final class RecordingStream implements AutoCloseable, EventStream { public void startAsync() { PlatformRecording pr = PrivateAccess.getInstance().getPlatformRecording(recording); long startNanos = pr.start(); + updateOnCompleteHandler(); directoryStream.startAsync(startNanos); } @@ -380,4 +408,13 @@ public final class RecordingStream implements AutoCloseable, EventStream { public void onMetadata(Consumer action) { directoryStream.onMetadata(action); } + + private void updateOnCompleteHandler() { + if (maxAge != null || maxSize != 0) { + // User has set a chunk removal policy + directoryStream.setChunkCompleteHandler(null); + } else { + directoryStream.setChunkCompleteHandler(new ChunkConsumer(recording)); + } + } } diff --git a/src/jdk.jfr/share/classes/jdk/jfr/internal/consumer/ChunkParser.java b/src/jdk.jfr/share/classes/jdk/jfr/internal/consumer/ChunkParser.java index 043bac38589..4d2d94d5dcb 100644 --- a/src/jdk.jfr/share/classes/jdk/jfr/internal/consumer/ChunkParser.java +++ b/src/jdk.jfr/share/classes/jdk/jfr/internal/consumer/ChunkParser.java @@ -488,4 +488,16 @@ public final class ChunkParser { public boolean hasStaleMetadata() { return staleMetadata; } + + public void resetCache() { + LongMap ps = this.parsers; + if (ps != null) { + ps.forEach(p -> { + if (p instanceof EventParser) { + EventParser ep = (EventParser) p; + ep.resetCache(); + } + }); + } + } } diff --git a/src/jdk.jfr/share/classes/jdk/jfr/internal/consumer/EventDirectoryStream.java b/src/jdk.jfr/share/classes/jdk/jfr/internal/consumer/EventDirectoryStream.java index 192dabd8138..9a71c5c145c 100644 --- a/src/jdk.jfr/share/classes/jdk/jfr/internal/consumer/EventDirectoryStream.java +++ b/src/jdk.jfr/share/classes/jdk/jfr/internal/consumer/EventDirectoryStream.java @@ -159,6 +159,7 @@ public class EventDirectoryStream extends AbstractEventStream { } else { processUnordered(disp); } + currentParser.resetCache(); if (currentParser.getStartNanos() + currentParser.getChunkDuration() > filterEnd) { close(); return; diff --git a/src/jdk.jfr/share/classes/jdk/jfr/internal/consumer/EventFileStream.java b/src/jdk.jfr/share/classes/jdk/jfr/internal/consumer/EventFileStream.java index 6db1d7ae1bf..91c70973464 100644 --- a/src/jdk.jfr/share/classes/jdk/jfr/internal/consumer/EventFileStream.java +++ b/src/jdk.jfr/share/classes/jdk/jfr/internal/consumer/EventFileStream.java @@ -102,6 +102,7 @@ public final class EventFileStream extends AbstractEventStream { } else { processUnordered(disp); } + currentParser.resetCache(); if (isClosed() || currentParser.isLastChunk()) { return; }