8291519: jdk/jfr/api/event/TestShouldCommit.java failed with Unexpected value of shouldCommit()

Reviewed-by: mgronlun
This commit is contained in:
Erik Gahlin 2022-10-11 10:06:17 +00:00
parent 6053bf0f6a
commit fba763f825

View File

@ -1,5 +1,5 @@
/* /*
* Copyright (c) 2018, Oracle and/or its affiliates. All rights reserved. * Copyright (c) 2018, 2022, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
* *
* This code is free software; you can redistribute it and/or modify it * This code is free software; you can redistribute it and/or modify it
@ -23,171 +23,172 @@
package jdk.jfr.api.event; package jdk.jfr.api.event;
import java.nio.file.Path;
import java.time.Duration; import java.time.Duration;
import java.util.Set;
import jdk.jfr.Event; import jdk.jfr.Event;
import jdk.jfr.Recording; import jdk.jfr.Recording;
import jdk.test.lib.Asserts; import jdk.jfr.SettingControl;
import jdk.jfr.SettingDefinition;
import jdk.jfr.consumer.RecordingFile;
import static jdk.test.lib.Asserts.assertTrue;
import static jdk.test.lib.Asserts.assertFalse;
/** /**
* @test * @test
* @summary Test enable/disable event and verify recording has expected events. * @summary Test jdk.jfr.Event::shouldCommit()
* @key jfr * @key jfr
* @requires vm.hasJFR * @requires vm.hasJFR
* @library /test/lib * @library /test/lib
* @run main/othervm -Xlog:jfr+event+setting=trace jdk.jfr.api.event.TestShouldCommit * @run main/othervm jdk.jfr.api.event.TestShouldCommit
*/ */
public class TestShouldCommit { public class TestShouldCommit {
private static class CatEvent extends Event {
}
private static class DogEvent extends Event {
}
private static class BirdEvent extends Event {
public boolean isFlying;
@SettingDefinition
public boolean fly(FlySetting control) {
return control.shouldFly() == isFlying;
}
}
private static class FlySetting extends SettingControl {
private boolean shouldFly;
@Override
public String combine(Set<String> settingValues) {
for (String s : settingValues) {
if ("true".equals(s)) {
return "true";
}
}
return "false";
}
public boolean shouldFly() {
return shouldFly;
}
@Override
public void setValue(String settingValue) {
shouldFly = "true".equals(settingValue);
}
@Override
public String getValue() {
return String.valueOf(shouldFly);
}
}
public static void main(String[] args) throws Exception { public static void main(String[] args) throws Exception {
Recording rA = new Recording(); testEnablement();
testThreshold();
verifyShouldCommitFalse(); // No active recordings testCustomSetting();
testWithoutEnd();
rA.start(); testCommit();
rA.enable(MyEvent.class).withoutThreshold(); // recA=all
verifyShouldCommitTrue();
setThreshold(rA, 100); // recA=100
verifyThreshold(100);
setThreshold(rA, 200); // recA=200
verifyThreshold(200);
Recording rB = new Recording();
verifyThreshold(200); // recA=200, recB=not started
rB.start();
verifyThreshold(200); // recA=200, recB=not specified, settings from recA is used.
setThreshold(rB, 100); // recA=200, recB=100
verifyThreshold(100);
setThreshold(rB, 300); // recA=200, recB=300
verifyThreshold(200);
rA.disable(MyEvent.class); // recA=disabled, recB=300
verifyThreshold(300);
rB.disable(MyEvent.class); // recA=disabled, recB=disabled
verifyShouldCommitFalse();
setThreshold(rA, 200); // recA=200, recB=disabled
verifyThreshold(200);
rB.enable(MyEvent.class).withoutThreshold(); // recA=200, recB=all
verifyShouldCommitTrue();
setThreshold(rB, 100); // recA=200, recB=100
verifyThreshold(100);
rB.stop(); // recA=200, recB=stopped
verifyThreshold(200);
rA.stop(); // recA=stopped, recB=stopped
verifyShouldCommitFalse();
rA.close();
rB.close();
verifyShouldCommitFalse();
} }
private static void setThreshold(Recording r, long millis) { private static void testEnablement() throws Exception {
r.enable(MyEvent.class).withThreshold(Duration.ofMillis(millis)); DogEvent b = new DogEvent();
assertFalse(b.shouldCommit(), "Expected false before recording is started");
try (Recording r = new Recording()) {
r.enable(CatEvent.class);
r.disable(DogEvent.class);
r.start();
CatEvent c = new CatEvent();
assertTrue(c.shouldCommit(), "Expected true for enabled event");
DogEvent d = new DogEvent();
assertFalse(d.shouldCommit(), "Expected false for disabled event");
} }
private static void verifyThreshold(long threshold) throws Exception { CatEvent c = new CatEvent();
// Create 2 events, with different sleep time between begin() and end() assertFalse(c.shouldCommit(), "Expected false after recording is stopped");
// First event ends just before threshold, the other just after.
verifyThreshold(threshold-5, threshold);
verifyThreshold(threshold+5, threshold);
} }
private static void verifyThreshold(long sleepMs, long thresholdMs) throws Exception { private static void testThreshold() throws Exception {
MyEvent event = new MyEvent(); try (Recording r = new Recording()) {
r.enable(CatEvent.class).withThreshold(Duration.ofNanos(0));
r.enable(DogEvent.class).withThreshold(Duration.ofDays(1));
r.start();
long beforeStartNanos = System.nanoTime(); CatEvent c = new CatEvent();
event.begin(); c.begin();
long afterStartNanos = System.nanoTime();
Thread.sleep(sleepMs);
long beforeStopNanos = System.nanoTime();
event.end();
long afterStopNanos = System.nanoTime();
boolean actualShouldCommit = event.shouldCommit();
final long safetyMarginNanos = 2000000; // Allow an error of 2 ms. May have to be tuned.
//Duration of event has been at least minDurationMicros
long minDurationMicros = (beforeStopNanos - afterStartNanos - safetyMarginNanos) / 1000;
//Duration of event has been at most maxDurationMicros
long maxDurationMicros = (afterStopNanos - beforeStartNanos + safetyMarginNanos) / 1000;
Asserts.assertLessThanOrEqual(minDurationMicros, maxDurationMicros, "Wrong min/max duration. Test error.");
long thresholdMicros = thresholdMs * 1000;
Boolean shouldCommit = null;
if (minDurationMicros > thresholdMicros) {
shouldCommit = new Boolean(true); // shouldCommit() must be true
} else if (maxDurationMicros < thresholdMicros) {
shouldCommit = new Boolean(false); // shouldCommit() must be false
} else {
// Too close to call. No checks are done since we are not sure of expected shouldCommit().
}
System.out.printf(
"threshold=%d, duration=[%d-%d], shouldCommit()=%b, expected=%s%n",
thresholdMicros, minDurationMicros, maxDurationMicros, actualShouldCommit,
(shouldCommit!=null ? shouldCommit : "too close to call"));
try {
if (shouldCommit != null) {
Asserts.assertEquals(shouldCommit.booleanValue(), actualShouldCommit, "Wrong shouldCommit()");
}
} catch (Exception e) {
System.out.println("Unexpected value of shouldCommit(). Searching for active threshold...");
searchThreshold(thresholdMs, 2000+thresholdMs);
throw e;
}
}
// Sleeps until shouldCommit() is true, or give up. Used for logging.
private static void searchThreshold(long expectedMs, long maxMs) throws Exception {
long start = System.nanoTime();
long stop = start + maxMs * 1000000;
MyEvent event = new MyEvent();
event.begin();
event.end();
while (!event.shouldCommit() && System.nanoTime() < stop) {
Thread.sleep(1); Thread.sleep(1);
event.end(); c.end();
assertTrue(c.shouldCommit(), "Expected true if above threshold");
DogEvent d = new DogEvent();
d.begin();
Thread.sleep(1);
d.end();
assertFalse(d.shouldCommit(), "Expected false if below threshold");
} }
long durationMicros = (System.nanoTime() - start) / 1000;
long expectedMicros = expectedMs * 1000;
System.out.printf("shouldCommit()=%b after %,d ms, expected %,d%n", event.shouldCommit(), durationMicros, expectedMicros);
} }
private static void verifyShouldCommitFalse() { private static void testCustomSetting() throws Exception {
MyEvent event = new MyEvent(); try (Recording r = new Recording()) {
event.begin(); r.enable(BirdEvent.class).with("fly", "true");
event.end(); r.start();
Asserts.assertFalse(event.shouldCommit(), "shouldCommit() expected false"); BirdEvent b1 = new BirdEvent();
b1.isFlying = false;
b1.begin();
b1.end();
assertFalse(b1.shouldCommit(), "Expected false if rejected by custom setting");
BirdEvent b2 = new BirdEvent();
b2.isFlying = true;
b2.begin();
b2.end();
assertTrue(b2.shouldCommit(), "Expected true if accepted by custom setting");
}
} }
private static void verifyShouldCommitTrue() { private static void testWithoutEnd() throws Exception {
MyEvent event = new MyEvent(); try (Recording r = new Recording()) {
event.begin(); r.enable(CatEvent.class).withThreshold(Duration.ofDays(0));
event.end(); r.enable(DogEvent.class).withThreshold(Duration.ofDays(1));
Asserts.assertTrue(event.shouldCommit(), "shouldCommit() expected true"); r.start();
CatEvent c = new CatEvent();
c.begin();
Thread.sleep(1);
assertTrue(c.shouldCommit(), "Expected true when above threshold and end() not invoked");
DogEvent d = new DogEvent();
d.begin();
Thread.sleep(1);
assertFalse(d.shouldCommit(), "Expected false when below threshold and end() not invoked");
}
} }
private static class MyEvent extends Event { private static void testCommit() throws Exception {
try (Recording r = new Recording()) {
r.enable(CatEvent.class);
r.start();
CatEvent c = new CatEvent();
c.begin();
Thread.sleep(1);
c.end();
if (c.shouldCommit()) {
c.commit();
}
r.stop();
Path file = Path.of("dump.jfr");
r.dump(file);
boolean hasEvent = RecordingFile.readAllEvents(file).size() > 0;
assertTrue(hasEvent, "Expected event when using commit() after shouldCommit()");
}
} }
} }