Skip to content

Commit e80012e

Browse files
committedApr 13, 2021
8264768: JFR: Allow events to be printed to the log
Reviewed-by: mgronlun
1 parent 3b576ed commit e80012e

29 files changed

+1114
-31
lines changed
 

‎src/hotspot/share/jfr/jni/jfrJavaSupport.cpp

+17-15
Original file line numberDiff line numberDiff line change
@@ -488,25 +488,27 @@ Klass* JfrJavaSupport::klass(const jobject handle) {
488488
}
489489

490490
// caller needs ResourceMark
491-
const char* JfrJavaSupport::c_str(jstring string, Thread* t) {
491+
const char* JfrJavaSupport::c_str(oop string, Thread* t) {
492492
DEBUG_ONLY(check_java_thread_in_vm(t));
493-
if (string == NULL) {
494-
return NULL;
495-
}
496-
const char* temp = NULL;
497-
const oop java_string = resolve_non_null(string);
498-
const typeArrayOop value = java_lang_String::value(java_string);
493+
char* resource_copy = NULL;
494+
const typeArrayOop value = java_lang_String::value(string);
499495
if (value != NULL) {
500-
const size_t length = java_lang_String::utf8_length(java_string, value);
501-
temp = NEW_RESOURCE_ARRAY_IN_THREAD(t, const char, (length + 1));
502-
if (temp == NULL) {
503-
JfrJavaSupport::throw_out_of_memory_error("Unable to allocate thread local native memory", t);
504-
return NULL;
496+
const int length = java_lang_String::utf8_length(string, value);
497+
resource_copy = NEW_RESOURCE_ARRAY_IN_THREAD(t, char, (length + 1));
498+
if (resource_copy == NULL) {
499+
JfrJavaSupport::throw_out_of_memory_error("Unable to allocate thread local native memory", t);
500+
return NULL;
505501
}
506-
assert(temp != NULL, "invariant");
507-
java_lang_String::as_utf8_string(java_string, value, const_cast<char*>(temp), (int) length + 1);
502+
assert(resource_copy != NULL, "invariant");
503+
java_lang_String::as_utf8_string(string, value, resource_copy, length + 1);
508504
}
509-
return temp;
505+
return resource_copy;
506+
}
507+
508+
// caller needs ResourceMark
509+
const char* JfrJavaSupport::c_str(jstring string, Thread* t) {
510+
DEBUG_ONLY(check_java_thread_in_vm(t));
511+
return string != NULL ? c_str(resolve_non_null(string), t) : NULL;
510512
}
511513

512514
/*

‎src/hotspot/share/jfr/jni/jfrJavaSupport.hpp

+1
Original file line numberDiff line numberDiff line change
@@ -77,6 +77,7 @@ class JfrJavaSupport : public AllStatic {
7777
static Klass* klass(const jobject handle);
7878
// caller needs ResourceMark
7979
static const char* c_str(jstring string, Thread* jt);
80+
static const char* c_str(oop string, Thread* t);
8081

8182
// exceptions
8283
static void throw_illegal_state_exception(const char* message, TRAPS);

‎src/hotspot/share/jfr/jni/jfrJniMethod.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -258,6 +258,10 @@ JVM_ENTRY_NO_ENV(void, jfr_log(JNIEnv* env, jobject jvm, jint tag_set, jint leve
258258
JfrJavaLog::log(tag_set, level, message, thread);
259259
JVM_END
260260

261+
JVM_ENTRY_NO_ENV(void, jfr_log_event(JNIEnv* env, jobject jvm, jint level, jobjectArray lines, jboolean system))
262+
JfrJavaLog::log_event(env, level, lines, system == JNI_TRUE, thread);
263+
JVM_END
264+
261265
JVM_ENTRY_NO_ENV(void, jfr_subscribe_log_level(JNIEnv* env, jobject jvm, jobject log_tag, jint id))
262266
JfrJavaLog::subscribe_log_level(log_tag, id, thread);
263267
JVM_END

‎src/hotspot/share/jfr/jni/jfrJniMethod.hpp

+2
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,8 @@ void JNICALL jfr_subscribe_log_level(JNIEnv* env, jobject jvm, jobject log_tag,
7171

7272
void JNICALL jfr_log(JNIEnv* env, jobject jvm, jint tag_set, jint level, jstring message);
7373

74+
void JNICALL jfr_log_event(JNIEnv* env, jobject jvm, jint level, jobjectArray lines, jboolean system);
75+
7476
void JNICALL jfr_retransform_classes(JNIEnv* env, jobject jvm, jobjectArray classes);
7577

7678
void JNICALL jfr_set_enabled(JNIEnv* env, jobject jvm, jlong event_type_id, jboolean enabled);

‎src/hotspot/share/jfr/jni/jfrJniMethodRegistration.cpp

+1
Original file line numberDiff line numberDiff line change
@@ -51,6 +51,7 @@ JfrJniMethodRegistration::JfrJniMethodRegistration(JNIEnv* env) {
5151
(char*)"getTicksFrequency", (char*)"()J", (void*)jfr_elapsed_frequency,
5252
(char*)"subscribeLogLevel", (char*)"(Ljdk/jfr/internal/LogTag;I)V", (void*)jfr_subscribe_log_level,
5353
(char*)"log", (char*)"(IILjava/lang/String;)V", (void*)jfr_log,
54+
(char*)"logEvent", (char*)"(I[Ljava/lang/String;Z)V", (void*)jfr_log_event,
5455
(char*)"retransformClasses", (char*)"([Ljava/lang/Class;)V", (void*)jfr_retransform_classes,
5556
(char*)"setEnabled", (char*)"(JZ)V", (void*)jfr_set_enabled,
5657
(char*)"setFileNotification", (char*)"(J)V", (void*)jfr_set_file_notification,

‎src/hotspot/share/jfr/periodic/jfrNetworkUtilization.cpp

-1
Original file line numberDiff line numberDiff line change
@@ -156,7 +156,6 @@ void JfrNetworkUtilization::send_events() {
156156
if (!get_interfaces(&network_interfaces)) {
157157
return;
158158
}
159-
log_trace(jfr, event)("Reporting network utilization");
160159
static JfrTicks last_sample_instant;
161160
const JfrTicks cur_time = JfrTicks::now();
162161
if (cur_time > last_sample_instant) {

‎src/hotspot/share/jfr/utilities/jfrJavaLog.cpp

+34
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,9 @@
2828
#include "jfr/utilities/jfrLogTagSets.hpp"
2929
#include "logging/log.hpp"
3030
#include "logging/logConfiguration.hpp"
31+
#include "logging/logMessage.hpp"
3132
#include "memory/resourceArea.hpp"
33+
#include "oops/objArrayOop.inline.hpp"
3234
#include "runtime/thread.inline.hpp"
3335

3436
#define JFR_LOG_TAGS_CONCATED(T0, T1, T2, T3, T4, T5, ...) \
@@ -119,6 +121,38 @@ void JfrJavaLog::subscribe_log_level(jobject log_tag, jint id, TRAPS) {
119121
}
120122
}
121123

124+
void JfrJavaLog::log_event(JNIEnv* env, jint level, jobjectArray lines, bool system, TRAPS) {
125+
DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_vm(THREAD));
126+
if (lines == NULL) {
127+
return;
128+
}
129+
if (level < (jint)LogLevel::First || level > (jint)LogLevel::Last) {
130+
JfrJavaSupport::throw_illegal_argument_exception("LogLevel passed is outside valid range", THREAD);
131+
return;
132+
}
133+
134+
objArrayOop the_lines = objArrayOop(JfrJavaSupport::resolve_non_null(lines));
135+
assert(the_lines != NULL, "invariant");
136+
assert(the_lines->is_array(), "must be array");
137+
const int length = the_lines->length();
138+
139+
ResourceMark rm(THREAD);
140+
LogMessage(jfr, event) jfr_event;
141+
LogMessage(jfr, system, event) jfr_event_system;
142+
for (int i = 0; i < length; ++i) {
143+
const char* text = JfrJavaSupport::c_str(the_lines->obj_at(i), THREAD);
144+
if (text == NULL) {
145+
// An oome has been thrown and is pending.
146+
return;
147+
}
148+
if (system) {
149+
jfr_event_system.write((LogLevelType)level, "%s", text);
150+
} else {
151+
jfr_event.write((LogLevelType)level, "%s", text);
152+
}
153+
}
154+
}
155+
122156
void JfrJavaLog::log(jint tag_set, jint level, jstring message, TRAPS) {
123157
DEBUG_ONLY(JfrJavaSupport::check_java_thread_in_vm(THREAD));
124158
if (message == NULL) {

‎src/hotspot/share/jfr/utilities/jfrJavaLog.hpp

+1
Original file line numberDiff line numberDiff line change
@@ -43,6 +43,7 @@ class JfrJavaLog : public AllStatic {
4343
public:
4444
static void subscribe_log_level(jobject log_tag, jint id, TRAPS);
4545
static void log(jint tag_set, jint level, jstring message, TRAPS);
46+
static void log_event(JNIEnv* env, jint level, jobjectArray lines, bool system, TRAPS);
4647
};
4748

4849
#endif // SHARE_JFR_UTILITIES_JFRJAVALOG_HPP

‎src/jdk.jfr/share/classes/jdk/jfr/internal/JVM.java

+11
Original file line numberDiff line numberDiff line change
@@ -188,6 +188,17 @@ private JVM() {
188188
*/
189189
public static native void log(int tagSetId, int level, String message);
190190

191+
/**
192+
* Log an event to jfr+event or jfr+event+system.
193+
* <p>
194+
* Caller should ensure that message is not null or too large to handle.
195+
*
196+
* @param level log level
197+
* @param lines lines to log
198+
* @param system if lines should be written to jfr+event+system
199+
*/
200+
public static native void logEvent(int level, String[] lines, boolean system);
201+
191202
/**
192203
* Subscribe to LogLevel updates for LogTag
193204
*

‎src/jdk.jfr/share/classes/jdk/jfr/internal/LogTag.java

+10-1
Original file line numberDiff line numberDiff line change
@@ -95,4 +95,13 @@ public enum LogTag {
9595
LogTag(int tagId) {
9696
id = tagId;
9797
}
98-
}
98+
99+
public LogLevel level() {
100+
for (LogLevel l : LogLevel.values()) {
101+
if (l.level == tagSetLevel) {
102+
return l;
103+
}
104+
}
105+
return LogLevel.WARN; // default
106+
}
107+
}

‎src/jdk.jfr/share/classes/jdk/jfr/internal/Logger.java

+23-1
Original file line numberDiff line numberDiff line change
@@ -25,14 +25,17 @@
2525

2626
package jdk.jfr.internal;
2727

28+
import java.util.Arrays;
29+
2830
/**
2931
* JFR logger
3032
*
3133
*/
3234

3335
public final class Logger {
3436

35-
private static final int MAX_SIZE = 10000;
37+
private static final int MAX_SIZE = 10_000;
38+
private static final int MAX_EVENT_SIZE = 100_000;
3639
static {
3740
// This will try to initialize the JVM logging system
3841
JVMSupport.tryToInitializeJVM();
@@ -45,6 +48,25 @@ public static void log(LogTag logTag, LogLevel logLevel, String message) {
4548
}
4649
}
4750

51+
public static void logEvent(LogLevel logLevel, String[] lines, boolean system) {
52+
if (lines == null || lines.length == 0) {
53+
return;
54+
}
55+
if (shouldLog(LogTag.JFR_EVENT, logLevel) || shouldLog(LogTag.JFR_SYSTEM_EVENT, logLevel)) {
56+
int size = 0;
57+
for (int i = 0; i < lines.length; i++) {
58+
String line = lines[i];
59+
if (size + line.length() > MAX_EVENT_SIZE) {
60+
lines = Arrays.copyOf(lines, i + 1);
61+
lines[i] = "...";
62+
break;
63+
}
64+
size+=line.length();
65+
}
66+
JVM.logEvent(logLevel.level, lines, system);
67+
}
68+
}
69+
4870
private static void logInternal(LogTag logTag, LogLevel logLevel, String message) {
4971
if (message == null || message.length() < MAX_SIZE) {
5072
JVM.log(logTag.id, logLevel.level, message);

‎src/jdk.jfr/share/classes/jdk/jfr/internal/PlatformEventType.java

+6-2
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,7 @@
3939
*/
4040
public final class PlatformEventType extends Type {
4141
private final boolean isJVM;
42-
private final boolean isJDK;
42+
private final boolean isJDK;
4343
private final boolean isMethodSampling;
4444
private final List<SettingDescriptor> settings = new ArrayList<>(5);
4545
private final boolean dynamicSettings;
@@ -188,6 +188,10 @@ public boolean isEnabled() {
188188
return enabled;
189189
}
190190

191+
public boolean isSystem() {
192+
return isJVM || isJDK;
193+
}
194+
191195
public boolean isJVM() {
192196
return isJVM;
193197
}
@@ -285,7 +289,7 @@ public boolean setRegistered(boolean registered) {
285289
if (this.registered != registered) {
286290
this.registered = registered;
287291
updateCommittable();
288-
LogTag logTag = isJVM() || isJDK() ? LogTag.JFR_SYSTEM_EVENT : LogTag.JFR_EVENT;
292+
LogTag logTag = isSystem() ? LogTag.JFR_SYSTEM_METADATA : LogTag.JFR_METADATA;
289293
if (registered) {
290294
Logger.log(logTag, LogLevel.INFO, "Registered " + getLogName());
291295
} else {

‎src/jdk.jfr/share/classes/jdk/jfr/internal/PlatformRecorder.java

+26-1
Original file line numberDiff line numberDiff line change
@@ -57,12 +57,13 @@
5757
import jdk.jfr.events.ActiveSettingEvent;
5858
import jdk.jfr.internal.SecuritySupport.SafePath;
5959
import jdk.jfr.internal.SecuritySupport.SecureRecorderListener;
60+
import jdk.jfr.internal.consumer.EventLog;
6061
import jdk.jfr.internal.instrument.JDKEvents;
6162

6263
public final class PlatformRecorder {
6364

6465

65-
private final List<PlatformRecording> recordings = new ArrayList<>();
66+
private final ArrayList<PlatformRecording> recordings = new ArrayList<>();
6667
private static final List<SecureRecorderListener> changeListeners = new ArrayList<>();
6768
private final Repository repository;
6869
private static final JVM jvm = JVM.getJVM();
@@ -245,6 +246,9 @@ synchronized long start(PlatformRecording recording) {
245246
RepositoryChunk newChunk = null;
246247
if (toDisk) {
247248
newChunk = repository.newChunk(zdtNow);
249+
if (EventLog.shouldLog()) {
250+
EventLog.start();
251+
}
248252
MetadataRepository.getInstance().setOutput(newChunk.getFile().toString());
249253
} else {
250254
MetadataRepository.getInstance().setOutput(null);
@@ -259,6 +263,9 @@ synchronized long start(PlatformRecording recording) {
259263
RepositoryChunk newChunk = null;
260264
if (toDisk) {
261265
newChunk = repository.newChunk(zdtNow);
266+
if (EventLog.shouldLog()) {
267+
EventLog.start();
268+
}
262269
RequestEngine.doChunkEnd();
263270
MetadataRepository.getInstance().setOutput(newChunk.getFile().toString());
264271
startNanos = jvm.getChunkStartNanos();
@@ -347,6 +354,9 @@ synchronized void stop(PlatformRecording recording) {
347354
RequestEngine.setFlushInterval(Long.MAX_VALUE);
348355
}
349356
recording.setState(RecordingState.STOPPED);
357+
if (!isToDisk()) {
358+
EventLog.stop();
359+
}
350360
}
351361

352362
private void dumpMemoryToDestination(PlatformRecording recording) {
@@ -476,13 +486,28 @@ private void periodicTask() {
476486
if (jvm.shouldRotateDisk()) {
477487
rotateDisk();
478488
}
489+
if (isToDisk()) {
490+
EventLog.update();
491+
}
479492
}
480493
long minDelta = RequestEngine.doPeriodic();
481494
long wait = Math.min(minDelta, Options.getWaitInterval());
482495
takeNap(wait);
483496
}
484497
}
485498

499+
private boolean isToDisk() {
500+
// Use indexing to avoid Iterator allocation if nothing happens
501+
int count = recordings.size();
502+
for (int i = 0; i < count; i++) {
503+
PlatformRecording r = recordings.get(i);
504+
if (r.isToDisk() && r.getState() == RecordingState.RUNNING) {
505+
return true;
506+
}
507+
}
508+
return false;
509+
}
510+
486511
private void takeNap(long duration) {
487512
try {
488513
synchronized (JVM.FILE_DELTA_CHANGE) {

‎src/jdk.jfr/share/classes/jdk/jfr/internal/RequestEngine.java

+6-6
Original file line numberDiff line numberDiff line change
@@ -68,15 +68,15 @@ private void execute() {
6868
} else {
6969
jvm.emitEvent(type.getId(), JVM.counterTime(), 0);
7070
}
71-
if (Logger.shouldLog(LogTag.JFR_SYSTEM_EVENT, LogLevel.DEBUG)) {
72-
Logger.log(LogTag.JFR_SYSTEM_EVENT, LogLevel.DEBUG, "Executed periodic hook for " + type.getLogName());
71+
if (Logger.shouldLog(LogTag.JFR_SYSTEM, LogLevel.DEBUG)) {
72+
Logger.log(LogTag.JFR_SYSTEM, LogLevel.DEBUG, "Executed periodic hook for " + type.getLogName());
7373
}
7474
} else {
7575
executeSecure();
7676
}
7777
} catch (Throwable e) {
7878
// Prevent malicious user to propagate exception callback in the wrong context
79-
Logger.log(LogTag.JFR_SYSTEM_EVENT, LogLevel.WARN, "Exception occurred during execution of period hook for " + type.getLogName());
79+
Logger.log(LogTag.JFR_SYSTEM, LogLevel.WARN, "Exception occurred during execution of period hook for " + type.getLogName());
8080
}
8181
}
8282

@@ -137,10 +137,10 @@ public static void addTrustedJDKHook(Class<? extends Event> eventClass, Runnable
137137
}
138138

139139
private static void logHook(String action, PlatformEventType type) {
140-
if (type.isJDK() || type.isJVM()) {
141-
Logger.log(LogTag.JFR_SYSTEM_EVENT, LogLevel.INFO, action + " periodic hook for " + type.getLogName());
140+
if (type.isSystem()) {
141+
Logger.log(LogTag.JFR_SYSTEM, LogLevel.INFO, action + " periodic hook for " + type.getLogName());
142142
} else {
143-
Logger.log(LogTag.JFR_EVENT, LogLevel.INFO, action + " periodic hook for " + type.getLogName());
143+
Logger.log(LogTag.JFR, LogLevel.INFO, action + " periodic hook for " + type.getLogName());
144144
}
145145
}
146146

‎src/jdk.jfr/share/classes/jdk/jfr/internal/SecuritySupport.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -469,7 +469,7 @@ public static Thread createThreadWitNoPermissions(String threadName, Runnable ru
469469
return doPrivilegedWithReturn(() -> new Thread(runnable, threadName), new Permission[0]);
470470
}
471471

472-
static void setDaemonThread(Thread t, boolean daemon) {
472+
public static void setDaemonThread(Thread t, boolean daemon) {
473473
doPrivileged(()-> t.setDaemon(daemon), new RuntimePermission("modifyThread"));
474474
}
475475

‎src/jdk.jfr/share/classes/jdk/jfr/internal/consumer/AbstractEventStream.java

+8
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,8 @@ public abstract class AbstractEventStream implements EventStream {
6666

6767
private volatile boolean closed;
6868

69+
private boolean daemon = false;
70+
6971
AbstractEventStream(AccessControlContext acc, PlatformRecording recording, List<Configuration> configurations) throws IOException {
7072
this.accessControllerContext = Objects.requireNonNull(acc);
7173
this.recording = recording;
@@ -101,6 +103,11 @@ public final void setReuse(boolean reuse) {
101103
streamConfiguration.setReuse(reuse);
102104
}
103105

106+
// Only used if -Xlog:jfr+event* is specified
107+
public final void setDaemon(boolean daemon) {
108+
this.daemon = daemon;
109+
}
110+
104111
@Override
105112
public final void setStartTime(Instant startTime) {
106113
Objects.nonNull(startTime);
@@ -219,6 +226,7 @@ public final void startAsync(long startNanos) {
219226
startInternal(startNanos);
220227
Runnable r = () -> run(accessControllerContext);
221228
thread = SecuritySupport.createThreadWitNoPermissions(nextThreadName(), r);
229+
SecuritySupport.setDaemonThread(thread, daemon);
222230
thread.start();
223231
}
224232

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,193 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation. Oracle designates this
8+
* particular file as subject to the "Classpath" exception as provided
9+
* by Oracle in the LICENSE file that accompanied this code.
10+
*
11+
* This code is distributed in the hope that it will be useful, but WITHOUT
12+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14+
* version 2 for more details (a copy is included in the LICENSE file that
15+
* accompanied this code).
16+
*
17+
* You should have received a copy of the GNU General Public License version
18+
* 2 along with this work; if not, write to the Free Software Foundation,
19+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20+
*
21+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22+
* or visit www.oracle.com if you need additional information or have any
23+
* questions.
24+
*/
25+
package jdk.jfr.internal.consumer;
26+
27+
import java.io.IOException;
28+
import java.io.PrintWriter;
29+
import java.io.Writer;
30+
import java.util.HashSet;
31+
import java.util.Set;
32+
33+
import jdk.jfr.EventType;
34+
import jdk.jfr.FlightRecorder;
35+
import jdk.jfr.consumer.EventStream;
36+
import jdk.jfr.consumer.RecordedEvent;
37+
import jdk.jfr.internal.LogLevel;
38+
import jdk.jfr.internal.LogTag;
39+
import jdk.jfr.internal.Logger;
40+
import jdk.jfr.internal.PlatformEventType;
41+
import jdk.jfr.internal.PrivateAccess;
42+
import jdk.jfr.internal.tool.PrettyWriter;
43+
44+
45+
public final class EventLog {
46+
private static final class LogWriter extends Writer {
47+
private final StringBuilder builder = new StringBuilder();
48+
private LogLevel level = LogLevel.WARN;
49+
private boolean system;
50+
51+
public void setSystem(boolean system) {
52+
this.system = system;
53+
}
54+
55+
public void setLevel(LogLevel level) {
56+
this.level = level;
57+
}
58+
59+
@Override
60+
public void write(char[] buffer, int off, int len) throws IOException {
61+
builder.append(buffer, off, len);
62+
}
63+
64+
@Override
65+
public void flush() throws IOException {
66+
String[] lines = builder.toString().split("\n");
67+
builder.setLength(0);
68+
Logger.logEvent(level, lines, system);
69+
}
70+
71+
@Override
72+
public void close() throws IOException {
73+
// ignore
74+
}
75+
}
76+
77+
private static EventStream logStream;
78+
private static PrettyWriter prettyWriter;
79+
private static LogWriter logWriter;
80+
private static Set<Long> systemEventLookup;
81+
82+
// 1) At least one disk recording must be running
83+
// before calling this method
84+
// 2) Caller must hold PlatformRecorder lock
85+
public static void update() {
86+
boolean shouldLog = shouldLog();
87+
if (shouldLog && !isLogging()) {
88+
start();
89+
return;
90+
}
91+
if (!shouldLog && isLogging()) {
92+
stop();
93+
return;
94+
}
95+
}
96+
97+
// 1) At least one disk recording must be running
98+
// before calling this method
99+
// 2) Caller must hold PlatformRecorder lock
100+
public static void start() {
101+
if (logStream != null) {
102+
return;
103+
}
104+
try {
105+
ensureSystemEventLookup();
106+
logStream = EventStream.openRepository();
107+
((AbstractEventStream)logStream).setDaemon(true);
108+
logStream.onEvent(EventLog::log);
109+
logWriter = new LogWriter();
110+
prettyWriter = new PrettyWriter(new PrintWriter(logWriter));
111+
prettyWriter.setLineSeparator("\n");
112+
logStream.startAsync();
113+
Logger.log(LogTag.JFR_SYSTEM, LogLevel.DEBUG, "Log stream started");
114+
} catch (Exception e) {
115+
Logger.log(LogTag.JFR_SYSTEM, LogLevel.WARN, "Unable to print events to the log");
116+
}
117+
}
118+
119+
// Caller must hold PlatformRecorder lock
120+
public static void stop() {
121+
if (logStream == null) {
122+
return;
123+
}
124+
try {
125+
logStream.close();
126+
logStream = null;
127+
Logger.log(LogTag.JFR_SYSTEM, LogLevel.DEBUG, "Log stream stopped");
128+
} catch (Exception e) {
129+
Logger.log(LogTag.JFR_SYSTEM, LogLevel.WARN, "Unable to stop printing events to the log");
130+
}
131+
}
132+
133+
public static boolean shouldLog() {
134+
if (Logger.shouldLog(LogTag.JFR_EVENT, LogLevel.DEBUG)) {
135+
return true;
136+
}
137+
if (Logger.shouldLog(LogTag.JFR_SYSTEM_EVENT, LogLevel.DEBUG)) {
138+
return true;
139+
}
140+
return false;
141+
}
142+
143+
private static void log(RecordedEvent event) {
144+
boolean system = isSystemEvent(event);
145+
LogTag tag = system ? LogTag.JFR_SYSTEM_EVENT : LogTag.JFR_EVENT;
146+
LogLevel level = tag.level();
147+
if (Logger.shouldLog(tag, LogLevel.TRACE)) {
148+
log(event, 64, level, system);
149+
return;
150+
}
151+
152+
if (Logger.shouldLog(tag, LogLevel.DEBUG)) {
153+
log(event, 5, level, system);
154+
return;
155+
}
156+
}
157+
158+
private static void log(RecordedEvent event, int stackDepth, LogLevel level, boolean system) {
159+
logWriter.setSystem(system);
160+
logWriter.setLevel(level);
161+
prettyWriter.setStackDepth(stackDepth);
162+
prettyWriter.print(event);
163+
prettyWriter.flush(true);
164+
try {
165+
logWriter.flush();
166+
} catch (IOException e) {
167+
Logger.log(LogTag.JFR_SYSTEM, LogLevel.WARN, "Unable to print event to the log");
168+
}
169+
}
170+
171+
private static boolean isSystemEvent(RecordedEvent event) {
172+
return systemEventLookup.contains(event.getEventType().getId());
173+
}
174+
175+
// The file format doesn't contains information if an event is
176+
// from the JDK/JVM, besides the prefix "jdk." which is not reliable
177+
// Therefore, create a lookup by event type ID to see if is a JDK/JVM event.
178+
private static void ensureSystemEventLookup() {
179+
if (systemEventLookup == null) {
180+
systemEventLookup = new HashSet<>();
181+
for (EventType type : FlightRecorder.getFlightRecorder().getEventTypes()) {
182+
PlatformEventType pe = PrivateAccess.getInstance().getPlatformEventType(type);
183+
if (pe.isSystem()) {
184+
systemEventLookup.add(pe.getId());
185+
}
186+
}
187+
}
188+
}
189+
190+
private static boolean isLogging() {
191+
return logStream != null;
192+
}
193+
}

‎src/jdk.jfr/share/classes/jdk/jfr/internal/tool/StructuredWriter.java

+6-3
Original file line numberDiff line numberDiff line change
@@ -28,8 +28,6 @@
2828
import java.io.PrintWriter;
2929

3030
abstract class StructuredWriter {
31-
private static final String LINE_SEPARATOR = String.format("%n");
32-
3331
private final PrintWriter out;
3432
private final StringBuilder builder = new StringBuilder(4000);
3533

@@ -38,11 +36,16 @@ abstract class StructuredWriter {
3836
private int column;
3937
// print first event immediately so tool feels responsive
4038
private boolean first = true;
39+
private String lineSeparator = String.format("%n");
4140

4241
StructuredWriter(PrintWriter p) {
4342
out = p;
4443
}
4544

45+
public void setLineSeparator(String lineSeparator) {
46+
this.lineSeparator = lineSeparator;
47+
}
48+
4649
protected final int getColumn() {
4750
return column;
4851
}
@@ -67,7 +70,7 @@ public final void printIndent() {
6770
}
6871

6972
public final void println() {
70-
builder.append(LINE_SEPARATOR);
73+
builder.append(lineSeparator);
7174
column = 0;
7275
}
7376

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,72 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
package jdk.jfr.api.consumer.log;
24+
25+
import java.io.IOException;
26+
import java.nio.file.Files;
27+
import java.nio.file.Path;
28+
29+
// Helper class for analyzing log output from a live process
30+
public class LogAnalyzer {
31+
private final Path path;
32+
33+
public LogAnalyzer(String filename) throws IOException {
34+
this.path = Path.of(filename);
35+
}
36+
37+
public void shouldNotContain(String text) throws Exception {
38+
System.out.println("Should not contain: '" + text + "'");
39+
while (true) {
40+
try {
41+
for (String line : Files.readAllLines(path)) {
42+
if (line.contains(text)) {
43+
throw new Exception("Found unexpected log message: " + line);
44+
}
45+
}
46+
return;
47+
} catch (IOException e) {
48+
System.out.println("Could not read log file " + path.toAbsolutePath());
49+
e.printStackTrace();
50+
}
51+
Thread.sleep(100);
52+
}
53+
}
54+
55+
public void await(String text) throws InterruptedException {
56+
System.out.println("Awaiting... '" + text + "' ");
57+
while (true) {
58+
try {
59+
for (String line : Files.readAllLines(path)) {
60+
if (line.contains(text)) {
61+
System.out.println("Found!");
62+
return;
63+
}
64+
}
65+
} catch (IOException e) {
66+
System.out.println("Could not read log file " + path.toAbsolutePath());
67+
e.printStackTrace();
68+
}
69+
Thread.sleep(100);
70+
}
71+
}
72+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,69 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
package jdk.jfr.api.consumer.log;
24+
25+
import java.time.Duration;
26+
27+
import jdk.jfr.Event;
28+
import jdk.jfr.FlightRecorder;
29+
import jdk.jfr.Recording;
30+
31+
/**
32+
* @test
33+
* @summary Tests that the event payload is printed when using
34+
* -Xlog:jfr+event=trace
35+
* @key jfr
36+
* @requires vm.hasJFR
37+
* @library /test/lib
38+
* @build jdk.jfr.api.consumer.log.LogAnalyzer
39+
* @run main/othervm
40+
* -Xlog:jfr+event=trace:file=test-content.log
41+
* jdk.jfr.api.consumer.log.TestContent
42+
*/
43+
public class TestContent {
44+
45+
public static class ApplicationEvent extends Event {
46+
int value;
47+
String text;
48+
boolean truth;
49+
}
50+
51+
public static void main(String... args) throws Exception {
52+
FlightRecorder.addPeriodicEvent(ApplicationEvent.class, () -> {
53+
ApplicationEvent e = new ApplicationEvent();
54+
e.value = 4711;
55+
e.text = "hello, world!";
56+
e.truth = true;
57+
e.commit();
58+
});
59+
LogAnalyzer la = new LogAnalyzer("test-content.log");
60+
try (Recording r = new Recording()) {
61+
r.enable(ApplicationEvent.class).withPeriod(Duration.ofSeconds(1));
62+
r.start();
63+
la.await("value = 4711");
64+
la.await("text = \"hello, world!\"");
65+
la.await("truth = true");
66+
r.stop();
67+
}
68+
}
69+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
package jdk.jfr.api.consumer.log;
24+
25+
import jdk.jfr.Recording;
26+
27+
/**
28+
* @test
29+
* @summary Tests that event logging can't be turned on and off
30+
* @key jfr
31+
* @requires vm.hasJFR
32+
* @library /test/lib
33+
* @build jdk.jfr.api.consumer.log.LogAnalyzer
34+
* @run main/othervm
35+
* -Xlog:jfr+event*=debug,jfr+system=debug:file=disk-on-off.log
36+
* jdk.jfr.api.consumer.log.TestDiskOnOff
37+
*/
38+
public class TestDiskOnOff {
39+
40+
public static void main(String... args) throws Exception {
41+
LogAnalyzer la = new LogAnalyzer("disk-on-off.log");
42+
try (Recording r = new Recording()) {
43+
r.start();
44+
la.await("Log stream started");
45+
}
46+
la.await("Log stream stopped");
47+
try (Recording r = new Recording()) {
48+
r.start();
49+
la.await("Log stream started");
50+
}
51+
la.await("Log stream stopped");
52+
}
53+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
package jdk.jfr.api.consumer.log;
24+
25+
import java.time.Duration;
26+
27+
import jdk.jfr.Event;
28+
import jdk.jfr.FlightRecorder;
29+
import jdk.jfr.Recording;
30+
import jdk.test.lib.dcmd.CommandExecutor;
31+
import jdk.test.lib.dcmd.PidJcmdExecutor;
32+
import jdk.test.lib.process.OutputAnalyzer;
33+
34+
/**
35+
* @test
36+
* @summary Tests that log responds to log level changes after JVM has
37+
* started
38+
* @key jfr
39+
* @requires vm.hasJFR
40+
* @library /test/lib
41+
* @build jdk.jfr.api.consumer.log.LogAnalyzer
42+
* @run main/othervm jdk.jfr.api.consumer.log.TestDynamicStart
43+
*/
44+
public class TestDynamicStart {
45+
private static final String FILE = "log.txt";
46+
47+
static class UserEvent extends Event {
48+
String message;
49+
}
50+
51+
public static void main(String... args) throws Exception {
52+
FlightRecorder.addPeriodicEvent(UserEvent.class, () -> {
53+
UserEvent event = new UserEvent();
54+
event.message = "Giraffe";
55+
event.commit();
56+
});
57+
LogAnalyzer la = new LogAnalyzer(FILE);
58+
try (Recording r = new Recording()) {
59+
r.enable(UserEvent.class).withPeriod(Duration.ofSeconds(1));
60+
r.start();
61+
executeJcmd("VM.log what=jfr+event=debug,jfr+system=debug output=" + FILE);
62+
la.await("Log stream started");
63+
la.await("Giraffe");
64+
executeJcmd("VM.log what=jfr+event=warning output=" + FILE);
65+
la.await("Log stream stopped");
66+
}
67+
}
68+
69+
private static void executeJcmd(String cmd) {
70+
CommandExecutor executor = new PidJcmdExecutor();
71+
OutputAnalyzer oa = executor.execute(cmd);
72+
oa.shouldHaveExitValue(0);
73+
}
74+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
package jdk.jfr.api.consumer.log;
24+
25+
import jdk.jfr.Recording;
26+
27+
/**
28+
* @test
29+
* @summary Tests that transition between disk=true/false works
30+
* @key jfr
31+
* @requires vm.hasJFR
32+
* @library /test/lib
33+
* @build jdk.jfr.api.consumer.log.LogAnalyzer
34+
* @run main/othervm
35+
* -Xlog:jfr+event*=debug,jfr+system=debug:file=memory-disk.log
36+
* jdk.jfr.api.consumer.log.TestMemoryDiskTransition
37+
*/
38+
public class TestMemoryDiskTransition {
39+
40+
public static void main(String... args) throws Exception {
41+
LogAnalyzer la = new LogAnalyzer("memory-disk.log");
42+
try (Recording r1 = new Recording()) {
43+
r1.setToDisk(false);
44+
r1.start();
45+
la.shouldNotContain("Log stream started");
46+
try (Recording r2 = new Recording()) {
47+
r2.setToDisk(true);
48+
r2.start(); // transition to disk
49+
la.shouldNotContain("Log stream stopped");
50+
la.await("Log stream started");
51+
r2.stop(); // transition back to memory
52+
}
53+
la.await("Log stream stopped");
54+
}
55+
}
56+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
package jdk.jfr.api.consumer.log;
24+
25+
import jdk.jfr.Recording;
26+
27+
/**
28+
* @test
29+
* @summary Tests that a stream is not started if disk=false
30+
* @key jfr
31+
* @requires vm.hasJFR
32+
* @library /test/lib
33+
* @build jdk.jfr.api.consumer.log.LogAnalyzer
34+
* @run main/othervm
35+
* -Xlog:jfr+event*=debug,jfr+system=debug:file=memory-only.log
36+
* jdk.jfr.api.consumer.log.TestMemoryOnly
37+
*/
38+
public class TestMemoryOnly {
39+
40+
public static void main(String... args) throws Exception {
41+
LogAnalyzer la = new LogAnalyzer("memory-only.log");
42+
try (Recording r = new Recording()) {
43+
r.setToDisk(false);
44+
r.start();
45+
r.stop();
46+
la.shouldNotContain("Log stream started");
47+
}
48+
la.shouldNotContain("Log stream stopped");
49+
}
50+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,59 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
package jdk.jfr.api.consumer.log;
24+
25+
import jdk.jfr.Event;
26+
import jdk.jfr.FlightRecorder;
27+
import jdk.jfr.Name;
28+
import jdk.jfr.Period;
29+
30+
/**
31+
* @test
32+
* @summary Tests that only system events are emitted
33+
* @key jfr
34+
* @requires vm.hasJFR
35+
* @library /test/lib
36+
* @build jdk.jfr.api.consumer.log.LogAnalyzer
37+
* @run main/othervm
38+
* -Xlog:jfr+event+system=trace:file=system.log
39+
* -XX:StartFlightRecording jdk.jfr.api.consumer.log.TestSystemEvents
40+
*/
41+
public class TestSystemEvents {
42+
// Testing with -XX:StartFlightRecording, since it's
43+
// a likely use case and there could be issues
44+
// with starting the stream before main.
45+
@Period("1 s")
46+
@Name("UserDefined")
47+
static class UserEvent extends Event {
48+
}
49+
50+
public static void main(String... args) throws Exception {
51+
FlightRecorder.addPeriodicEvent(UserEvent.class, () -> {
52+
UserEvent e = new UserEvent();
53+
e.commit();
54+
});
55+
LogAnalyzer la = new LogAnalyzer("system.log");
56+
la.await("CPULoad"); // emitted 1/s
57+
la.shouldNotContain("UserDefined");
58+
}
59+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,81 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
package jdk.jfr.api.consumer.log;
24+
25+
import jdk.jfr.Event;
26+
import jdk.jfr.FlightRecorder;
27+
import jdk.jfr.Period;
28+
import jdk.jfr.Recording;
29+
import jdk.jfr.StackTrace;
30+
31+
/**
32+
* @test
33+
* @summary Tests that large output is truncated
34+
* @key jfr
35+
* @requires vm.hasJFR
36+
* @library /test/lib
37+
* @build jdk.jfr.api.consumer.log.LogAnalyzer
38+
* @run main/othervm
39+
* -Xlog:jfr+event*=debug:file=truncate.log
40+
* jdk.jfr.api.consumer.log.TestTruncation
41+
*/
42+
public class TestTruncation {
43+
44+
@StackTrace(false) // Avoids '...' from stack trace truncation
45+
@Period("1 s")
46+
static class LargeEvent extends Event {
47+
String message1;
48+
String message2;
49+
String message3;
50+
String message4;
51+
String message5;
52+
String message6;
53+
String message7;
54+
String message8;
55+
String message9;
56+
String message10;
57+
}
58+
59+
public static void main(String... args) throws Exception {
60+
FlightRecorder.addPeriodicEvent(LargeEvent.class, () -> {
61+
String text = "#".repeat(10_000);
62+
LargeEvent event = new LargeEvent();
63+
event.message1 = text;
64+
event.message2 = text;
65+
event.message3 = text;
66+
event.message4 = text;
67+
event.message5 = text;
68+
event.message6 = text;
69+
event.message7 = text;
70+
event.message8 = text;
71+
event.message9 = text;
72+
event.message10 = text;
73+
event.commit();
74+
});
75+
LogAnalyzer la = new LogAnalyzer("truncate.log");
76+
try (Recording r = new Recording()) {
77+
r.start();
78+
la.await("...");
79+
}
80+
}
81+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,60 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
package jdk.jfr.api.consumer.log;
24+
25+
import jdk.jfr.Event;
26+
import jdk.jfr.FlightRecorder;
27+
import jdk.jfr.Name;
28+
import jdk.jfr.Period;
29+
30+
/**
31+
* @test
32+
* @summary Tests that only user events are emitted
33+
* @key jfr
34+
* @requires vm.hasJFR
35+
* @library /test/lib
36+
* @build jdk.jfr.api.consumer.log.LogAnalyzer
37+
* @run main/othervm
38+
* -Xlog:jfr+event=trace:file=user.log
39+
* -XX:StartFlightRecording
40+
* jdk.jfr.api.consumer.log.TestUserEvents
41+
*/
42+
public class TestUserEvents {
43+
// Testing with -XX:StartFlightRecording, since it's
44+
// a likely use case and there could be issues
45+
// with starting the stream before main.
46+
@Period("1 s")
47+
@Name("UserDefined")
48+
static class UserEvent extends Event {
49+
}
50+
51+
public static void main(String... args) throws Exception {
52+
FlightRecorder.addPeriodicEvent(UserEvent.class, () -> {
53+
UserEvent e = new UserEvent();
54+
e.commit();
55+
});
56+
LogAnalyzer la = new LogAnalyzer("user.log");
57+
la.await("UserDefined");
58+
la.shouldNotContain("CPULoad"); // Emitted 1/s
59+
}
60+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,112 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
package jdk.jfr.api.consumer.log;
24+
25+
import java.io.Closeable;
26+
import java.util.concurrent.Executors;
27+
import java.util.concurrent.TimeUnit;
28+
29+
import jdk.jfr.Event;
30+
import jdk.jfr.Name;
31+
32+
/**
33+
* @test
34+
* @summary Tests output from various tag sets and levels
35+
* @key jfr
36+
* @requires vm.hasJFR
37+
* @library /test/lib
38+
* @build jdk.jfr.api.consumer.log.LogAnalyzer
39+
* @run main/othervm
40+
* -Xlog:jfr+event*=trace:file=trace.log
41+
* -XX:StartFlightRecording
42+
* jdk.jfr.api.consumer.log.TestVerbosity trace
43+
* @run main/othervm
44+
* -Xlog:jfr+event*=debug:file=debug.log
45+
* -XX:StartFlightRecording
46+
* jdk.jfr.api.consumer.log.TestVerbosity debug
47+
* @run main/othervm
48+
* -Xlog:jfr+event*=info:file=info.log
49+
* -XX:StartFlightRecording
50+
* jdk.jfr.api.consumer.log.TestVerbosity info
51+
*/
52+
public class TestVerbosity {
53+
54+
@Name("UserDefined")
55+
static class UserEvent extends Event {
56+
}
57+
58+
public static void main(String... args) throws Exception {
59+
String level = args[0];
60+
var scheduler = Executors.newScheduledThreadPool(1);
61+
try (Closeable close = scheduler::shutdown) {
62+
63+
scheduler.scheduleAtFixedRate(() -> {
64+
method1();
65+
}, 0, 10, TimeUnit.MILLISECONDS);
66+
67+
LogAnalyzer la = new LogAnalyzer(level + ".log");
68+
System.out.println("Testing log level: " + level);
69+
if (level.equals("trace")) {
70+
la.await("CPULoad"); // Emitted 1/s
71+
la.await("UserDefined");
72+
la.await("method6");
73+
la.await("method1");
74+
}
75+
if (level.equals("debug")) {
76+
la.await("CPULoad");
77+
la.await("UserDefined");
78+
la.await("method6");
79+
la.shouldNotContain("method1");
80+
}
81+
if (level.equals("info")) {
82+
la.shouldNotContain("CPULoad");
83+
la.shouldNotContain("UserDefined");
84+
}
85+
}
86+
}
87+
88+
private static void method1() {
89+
method2();
90+
}
91+
92+
private static void method2() {
93+
method3();
94+
}
95+
96+
private static void method3() {
97+
method4();
98+
}
99+
100+
private static void method4() {
101+
method5();
102+
}
103+
104+
private static void method5() {
105+
method6();
106+
}
107+
108+
private static void method6() {
109+
UserEvent event = new UserEvent();
110+
event.commit();
111+
}
112+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,78 @@
1+
/*
2+
* Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
package jdk.jfr.api.consumer.log;
24+
25+
import java.io.Closeable;
26+
import java.time.Instant;
27+
import java.util.concurrent.CountDownLatch;
28+
import java.util.concurrent.Executors;
29+
import java.util.concurrent.TimeUnit;
30+
31+
import jdk.jfr.Event;
32+
import jdk.jfr.Name;
33+
import jdk.jfr.consumer.RecordingStream;
34+
35+
/**
36+
* @test
37+
* @summary Checks that it is possible to stream together with log stream
38+
* @key jfr
39+
* @requires vm.hasJFR
40+
* @library /test/lib
41+
* @build jdk.jfr.api.consumer.log.LogAnalyzer
42+
* @run main/othervm -Xlog:jfr+event*=debug:file=with-streaming.log
43+
* jdk.jfr.api.consumer.log.TestWithStreaming
44+
*/
45+
public class TestWithStreaming {
46+
47+
@Name("TwoStreams")
48+
static class TwoStreams extends Event {
49+
String message;
50+
}
51+
52+
public static void main(String... args) throws Exception {
53+
LogAnalyzer la = new LogAnalyzer("with-streaming.log");
54+
CountDownLatch latch = new CountDownLatch(2);
55+
try (RecordingStream rs = new RecordingStream()) {
56+
rs.enable(TwoStreams.class);
57+
rs.onEvent("TwoStreams", e -> {
58+
latch.countDown();
59+
});
60+
rs.setStartTime(Instant.MIN);
61+
rs.startAsync();
62+
TwoStreams e1 = new TwoStreams();
63+
e1.commit();
64+
TwoStreams e2 = new TwoStreams();
65+
e2.commit();
66+
latch.await();
67+
var scheduler = Executors.newScheduledThreadPool(1);
68+
try (Closeable close = scheduler::shutdown) {
69+
scheduler.scheduleAtFixedRate(() -> {
70+
TwoStreams e = new TwoStreams();
71+
e.message = "hello";
72+
e.commit();
73+
}, 0, 10, TimeUnit.MILLISECONDS);
74+
la.await("hello");
75+
}
76+
}
77+
}
78+
}

0 commit comments

Comments
 (0)
Please sign in to comment.