From ee2651b9e5a9ab468b4be73d43b8f643e9e92042 Mon Sep 17 00:00:00 2001 From: Jaroslav Bachorik Date: Fri, 21 May 2021 09:44:45 +0000 Subject: [PATCH] 8203359: Container level resources events Reviewed-by: sgehwolf, egahlin --- src/java.base/share/classes/module-info.java | 3 +- .../events/ContainerCPUThrottlingEvent.java | 57 ++++++++++ .../jfr/events/ContainerCPUUsageEvent.java | 59 ++++++++++ .../events/ContainerConfigurationEvent.java | 82 ++++++++++++++ .../jdk/jfr/events/ContainerIOUsageEvent.java | 55 +++++++++ .../jfr/events/ContainerMemoryUsageEvent.java | 58 ++++++++++ .../share/classes/jdk/jfr/internal/Utils.java | 20 ++++ .../jfr/internal/instrument/JDKEvents.java | 101 ++++++++++++++++- src/jdk.jfr/share/conf/jfr/default.jfc | 25 +++++ src/jdk.jfr/share/conf/jfr/profile.jfc | 25 +++++ .../jtreg/containers/docker/JfrReporter.java | 7 +- .../containers/docker/TestJFREvents.java | 104 ++++++++++++++++-- .../metadata/TestLookForUntestedEvents.java | 8 ++ test/lib/jdk/test/lib/jfr/EventNames.java | 6 + 14 files changed, 594 insertions(+), 16 deletions(-) create mode 100644 src/jdk.jfr/share/classes/jdk/jfr/events/ContainerCPUThrottlingEvent.java create mode 100644 src/jdk.jfr/share/classes/jdk/jfr/events/ContainerCPUUsageEvent.java create mode 100644 src/jdk.jfr/share/classes/jdk/jfr/events/ContainerConfigurationEvent.java create mode 100644 src/jdk.jfr/share/classes/jdk/jfr/events/ContainerIOUsageEvent.java create mode 100644 src/jdk.jfr/share/classes/jdk/jfr/events/ContainerMemoryUsageEvent.java diff --git a/src/java.base/share/classes/module-info.java b/src/java.base/share/classes/module-info.java index 7c5fa8d341b..e09ff7e8de8 100644 --- a/src/java.base/share/classes/module-info.java +++ b/src/java.base/share/classes/module-info.java @@ -218,7 +218,8 @@ module java.base { jdk.management.agent, jdk.internal.jvmstat; exports jdk.internal.platform to - jdk.management; + jdk.management, + jdk.jfr; exports jdk.internal.ref to java.desktop, jdk.incubator.foreign; diff --git a/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerCPUThrottlingEvent.java b/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerCPUThrottlingEvent.java new file mode 100644 index 00000000000..fe202a0ac3c --- /dev/null +++ b/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerCPUThrottlingEvent.java @@ -0,0 +1,57 @@ +/* + * Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2021, DataDog. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package jdk.jfr.events; + +import jdk.jfr.Category; +import jdk.jfr.DataAmount; +import jdk.jfr.Description; +import jdk.jfr.Enabled; +import jdk.jfr.Label; +import jdk.jfr.Name; +import jdk.jfr.Period; +import jdk.jfr.StackTrace; +import jdk.jfr.Threshold; +import jdk.jfr.Timespan; +import jdk.jfr.internal.Type; + +@Name(Type.EVENT_NAME_PREFIX + "ContainerCPUThrottling") +@Label("CPU Throttling") +@Category({"Operating System", "Processor"}) +@Description("Container CPU throttling related information") +public class ContainerCPUThrottlingEvent extends AbstractJDKEvent { + @Label("CPU Elapsed Slices") + @Description("Number of time-slice periods that have elapsed if a CPU quota has been setup for the container") + public long cpuElapsedSlices; + + @Label("CPU Throttled Slices") + @Description("Number of time-slice periods that the CPU has been throttled or limited due to exceeding CPU quota") + public long cpuThrottledSlices; + + @Label("CPU Throttled Time") + @Description("Total time duration, in nanoseconds, that the CPU has been throttled or limited due to exceeding CPU quota") + @Timespan + public long cpuThrottledTime; +} diff --git a/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerCPUUsageEvent.java b/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerCPUUsageEvent.java new file mode 100644 index 00000000000..d345bdc4217 --- /dev/null +++ b/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerCPUUsageEvent.java @@ -0,0 +1,59 @@ +/* + * Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2021, DataDog. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package jdk.jfr.events; + +import jdk.jfr.Category; +import jdk.jfr.DataAmount; +import jdk.jfr.Description; +import jdk.jfr.Enabled; +import jdk.jfr.Label; +import jdk.jfr.Name; +import jdk.jfr.Period; +import jdk.jfr.StackTrace; +import jdk.jfr.Threshold; +import jdk.jfr.Timespan; +import jdk.jfr.internal.Type; + +@Name(Type.EVENT_NAME_PREFIX + "ContainerCPUUsage") +@Label("CPU Usage") +@Category({"Operating System", "Processor"}) +@Description("Container CPU usage related information") +public class ContainerCPUUsageEvent extends AbstractJDKEvent { + @Label("CPU Time") + @Description("Aggregate time consumed by all tasks in the container") + @Timespan + public long cpuTime; + + @Label("CPU User Time") + @Description("Aggregate user time consumed by all tasks in the container") + @Timespan + public long cpuUserTime; + + @Label("CPU System Time") + @Description("Aggregate system time consumed by all tasks in the container") + @Timespan + public long cpuSystemTime; +} diff --git a/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerConfigurationEvent.java b/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerConfigurationEvent.java new file mode 100644 index 00000000000..0d20f1e7d97 --- /dev/null +++ b/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerConfigurationEvent.java @@ -0,0 +1,82 @@ +/* + * Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2021, DataDog. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package jdk.jfr.events; + +import jdk.jfr.Category; +import jdk.jfr.DataAmount; +import jdk.jfr.Description; +import jdk.jfr.Enabled; +import jdk.jfr.Label; +import jdk.jfr.Name; +import jdk.jfr.Period; +import jdk.jfr.StackTrace; +import jdk.jfr.Threshold; +import jdk.jfr.Timespan; +import jdk.jfr.internal.Type; + +@Name(Type.EVENT_NAME_PREFIX + "ContainerConfiguration") +@Label("Container Configuration") +@Category({"Operating System"}) +@Description("A set of container specific attributes") +public final class ContainerConfigurationEvent extends AbstractJDKEvent { + @Label("Container Type") + @Description("Container type information") + public String containerType; + + @Label("CPU Slice Period") + @Description("Length of the scheduling period for processes within the container") + @Timespan(Timespan.MICROSECONDS) + public long cpuSlicePeriod; + + @Label("CPU Quota") + @Description("Total available run-time allowed during each scheduling period for all tasks in the container") + @Timespan(Timespan.MICROSECONDS) + public long cpuQuota; + + @Label("CPU Shares") + @Description("Relative weighting of processes with the container used for prioritizing the scheduling of processes across " + + "all containers running on a host") + public long cpuShares; + + @Label("Effective CPU Count") + @Description("Number of effective processors that this container has available to it") + public long effectiveCpuCount; + + @Label("Memory Soft Limit") + @Description("Hint to the operating system that allows groups to specify the minimum required amount of physical memory") + @DataAmount + public long memorySoftLimit; + + @Label("Memory Limit") + @Description("Maximum amount of physical memory that can be allocated in the container") + @DataAmount + public long memoryLimit; + + @Label("Memory and Swap Limit") + @Description("Maximum amount of physical memory and swap space, in bytes, that can be allocated in the container") + @DataAmount + public long swapMemoryLimit; +} diff --git a/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerIOUsageEvent.java b/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerIOUsageEvent.java new file mode 100644 index 00000000000..fa3b73f7084 --- /dev/null +++ b/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerIOUsageEvent.java @@ -0,0 +1,55 @@ +/* + * Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2021, DataDog. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package jdk.jfr.events; + +import jdk.jfr.Category; +import jdk.jfr.DataAmount; +import jdk.jfr.Description; +import jdk.jfr.Enabled; +import jdk.jfr.Label; +import jdk.jfr.Name; +import jdk.jfr.Period; +import jdk.jfr.StackTrace; +import jdk.jfr.Threshold; +import jdk.jfr.Timespan; +import jdk.jfr.internal.Type; + + +@Name(Type.EVENT_NAME_PREFIX + "ContainerIOUsage") +@Label("Container IO Usage") +@Category({"Operating System", "File System"}) +@Description("Container IO usage related information") +public class ContainerIOUsageEvent extends AbstractJDKEvent { + + @Label("Block IO Request Count") + @Description("Number of block IO requests to the disk that have been issued by the container") + public long serviceRequests; + + @Label("Block IO Transfer") + @Description("Number of block IO bytes that have been transferred to/from the disk by the container") + @DataAmount + public long dataTransferred; +} diff --git a/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerMemoryUsageEvent.java b/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerMemoryUsageEvent.java new file mode 100644 index 00000000000..531f26aa9c5 --- /dev/null +++ b/src/jdk.jfr/share/classes/jdk/jfr/events/ContainerMemoryUsageEvent.java @@ -0,0 +1,58 @@ +/* + * Copyright (c) 2021, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2021, DataDog. All rights reserved. + * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. + * + * This code is free software; you can redistribute it and/or modify it + * under the terms of the GNU General Public License version 2 only, as + * published by the Free Software Foundation. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * This code is distributed in the hope that it will be useful, but WITHOUT + * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or + * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License + * version 2 for more details (a copy is included in the LICENSE file that + * accompanied this code). + * + * You should have received a copy of the GNU General Public License version + * 2 along with this work; if not, write to the Free Software Foundation, + * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. + * + * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA + * or visit www.oracle.com if you need additional information or have any + * questions. + */ +package jdk.jfr.events; + +import jdk.jfr.Category; +import jdk.jfr.DataAmount; +import jdk.jfr.Description; +import jdk.jfr.Enabled; +import jdk.jfr.Label; +import jdk.jfr.Name; +import jdk.jfr.Period; +import jdk.jfr.StackTrace; +import jdk.jfr.Threshold; +import jdk.jfr.Timespan; +import jdk.jfr.internal.Type; + +@Name(Type.EVENT_NAME_PREFIX + "ContainerMemoryUsage") +@Label("Container Memory Usage") +@Category({"Operating System", "Memory"}) +@Description("Container memory usage related information") +public final class ContainerMemoryUsageEvent extends AbstractJDKEvent { + @Label("Memory Fail Count") + @Description("Number of times that user memory requests in the container have exceeded the memory limit") + public long memoryFailCount; + + @Label("Memory Usage") + @Description("Amount of physical memory, in bytes, that is currently allocated in the current container") + @DataAmount + public long memoryUsage; + + @Label("Memory and Swap Usage") + @Description("Amount of physical memory and swap space, in bytes, that is currently allocated in the current container") + @DataAmount + public long swapMemoryUsage; +} diff --git a/src/jdk.jfr/share/classes/jdk/jfr/internal/Utils.java b/src/jdk.jfr/share/classes/jdk/jfr/internal/Utils.java index c37be2d7230..4cc5cde17b7 100644 --- a/src/jdk.jfr/share/classes/jdk/jfr/internal/Utils.java +++ b/src/jdk.jfr/share/classes/jdk/jfr/internal/Utils.java @@ -56,6 +56,7 @@ import java.util.Objects; import jdk.internal.org.objectweb.asm.ClassReader; import jdk.internal.org.objectweb.asm.util.CheckClassAdapter; +import jdk.internal.platform.Metrics; import jdk.jfr.Event; import jdk.jfr.FlightRecorderPermission; import jdk.jfr.Recording; @@ -90,6 +91,11 @@ public final class Utils { private static final int BASE = 10; private static long THROTTLE_OFF = -2; + /* + * This field will be lazily initialized and the access is not synchronized. + * The possible data race is benign and is worth of not introducing any contention here. + */ + private static Metrics[] metrics; public static void checkAccessFlightRecorder() throws SecurityException { SecurityManager sm = System.getSecurityManager(); @@ -720,6 +726,20 @@ public final class Utils { } } + public static boolean shouldSkipBytecode(String eventName, Class superClass) { + if (superClass.getClassLoader() != null || !superClass.getName().equals("jdk.jfr.events.AbstractJDKEvent")) { + return false; + } + return eventName.startsWith("jdk.Container") && getMetrics() == null; + } + + private static Metrics getMetrics() { + if (metrics == null) { + metrics = new Metrics[]{Metrics.systemMetrics()}; + } + return metrics[0]; + } + private static String formatPositiveDuration(Duration d){ if (d.compareTo(MICRO_SECOND) < 0) { // 0.000001 ms - 0.000999 ms diff --git a/src/jdk.jfr/share/classes/jdk/jfr/internal/instrument/JDKEvents.java b/src/jdk.jfr/share/classes/jdk/jfr/internal/instrument/JDKEvents.java index 50b67c51d53..89c9425cb9c 100644 --- a/src/jdk.jfr/share/classes/jdk/jfr/internal/instrument/JDKEvents.java +++ b/src/jdk.jfr/share/classes/jdk/jfr/internal/instrument/JDKEvents.java @@ -27,10 +27,16 @@ package jdk.jfr.internal.instrument; import java.util.ArrayList; import java.util.List; +import java.util.Map; import jdk.jfr.Event; import jdk.jfr.events.ActiveRecordingEvent; import jdk.jfr.events.ActiveSettingEvent; +import jdk.jfr.events.ContainerIOUsageEvent; +import jdk.jfr.events.ContainerConfigurationEvent; +import jdk.jfr.events.ContainerCPUUsageEvent; +import jdk.jfr.events.ContainerCPUThrottlingEvent; +import jdk.jfr.events.ContainerMemoryUsageEvent; import jdk.jfr.events.DirectBufferStatisticsEvent; import jdk.jfr.events.ErrorThrownEvent; import jdk.jfr.events.ExceptionStatisticsEvent; @@ -53,8 +59,10 @@ import jdk.jfr.internal.Logger; import jdk.jfr.internal.RequestEngine; import jdk.jfr.internal.SecuritySupport; -public final class JDKEvents { +import jdk.internal.platform.Container; +import jdk.internal.platform.Metrics; +public final class JDKEvents { private static final Class[] mirrorEventClasses = { DeserializationEvent.class, ProcessStartEvent.class, @@ -100,6 +108,12 @@ public final class JDKEvents { private static final JVM jvm = JVM.getJVM(); private static final Runnable emitExceptionStatistics = JDKEvents::emitExceptionStatistics; private static final Runnable emitDirectBufferStatistics = JDKEvents::emitDirectBufferStatistics; + private static final Runnable emitContainerConfiguration = JDKEvents::emitContainerConfiguration; + private static final Runnable emitContainerCPUUsage = JDKEvents::emitContainerCPUUsage; + private static final Runnable emitContainerCPUThrottling = JDKEvents::emitContainerCPUThrottling; + private static final Runnable emitContainerMemoryUsage = JDKEvents::emitContainerMemoryUsage; + private static final Runnable emitContainerIOUsage = JDKEvents::emitContainerIOUsage; + private static Metrics containerMetrics = null; private static boolean initializationTriggered; @SuppressWarnings("unchecked") @@ -112,9 +126,12 @@ public final class JDKEvents { for (Class eventClass : eventClasses) { SecuritySupport.registerEvent((Class) eventClass); } - initializationTriggered = true; + RequestEngine.addTrustedJDKHook(ExceptionStatisticsEvent.class, emitExceptionStatistics); RequestEngine.addTrustedJDKHook(DirectBufferStatisticsEvent.class, emitDirectBufferStatistics); + + initializeContainerEvents(); + initializationTriggered = true; } } catch (Exception e) { Logger.log(LogTag.JFR_SYSTEM, LogLevel.WARN, "Could not initialize JDK events. " + e.getMessage()); @@ -141,12 +158,84 @@ public final class JDKEvents { } } + private static void initializeContainerEvents() { + containerMetrics = Container.metrics(); + SecuritySupport.registerEvent(ContainerConfigurationEvent.class); + SecuritySupport.registerEvent(ContainerCPUUsageEvent.class); + SecuritySupport.registerEvent(ContainerCPUThrottlingEvent.class); + SecuritySupport.registerEvent(ContainerMemoryUsageEvent.class); + SecuritySupport.registerEvent(ContainerIOUsageEvent.class); + + RequestEngine.addTrustedJDKHook(ContainerConfigurationEvent.class, emitContainerConfiguration); + RequestEngine.addTrustedJDKHook(ContainerCPUUsageEvent.class, emitContainerCPUUsage); + RequestEngine.addTrustedJDKHook(ContainerCPUThrottlingEvent.class, emitContainerCPUThrottling); + RequestEngine.addTrustedJDKHook(ContainerMemoryUsageEvent.class, emitContainerMemoryUsage); + RequestEngine.addTrustedJDKHook(ContainerIOUsageEvent.class, emitContainerIOUsage); + } + private static void emitExceptionStatistics() { ExceptionStatisticsEvent t = new ExceptionStatisticsEvent(); t.throwables = ThrowableTracer.numThrowables(); t.commit(); } + private static void emitContainerConfiguration() { + if (containerMetrics != null) { + ContainerConfigurationEvent t = new ContainerConfigurationEvent(); + t.containerType = containerMetrics.getProvider(); + t.cpuSlicePeriod = containerMetrics.getCpuPeriod(); + t.cpuQuota = containerMetrics.getCpuQuota(); + t.cpuShares = containerMetrics.getCpuShares(); + t.effectiveCpuCount = containerMetrics.getEffectiveCpuCount(); + t.memorySoftLimit = containerMetrics.getMemorySoftLimit(); + t.memoryLimit = containerMetrics.getMemoryLimit(); + t.swapMemoryLimit = containerMetrics.getMemoryAndSwapLimit(); + t.commit(); + } + } + + private static void emitContainerCPUUsage() { + if (containerMetrics != null) { + ContainerCPUUsageEvent event = new ContainerCPUUsageEvent(); + + event.cpuTime = containerMetrics.getCpuUsage(); + event.cpuSystemTime = containerMetrics.getCpuSystemUsage(); + event.cpuUserTime = containerMetrics.getCpuUserUsage(); + event.commit(); + } + } + private static void emitContainerMemoryUsage() { + if (containerMetrics != null) { + ContainerMemoryUsageEvent event = new ContainerMemoryUsageEvent(); + + event.memoryFailCount = containerMetrics.getMemoryFailCount(); + event.memoryUsage = containerMetrics.getMemoryUsage(); + event.swapMemoryUsage = containerMetrics.getMemoryAndSwapUsage(); + event.commit(); + } + } + + private static void emitContainerIOUsage() { + if (containerMetrics != null) { + ContainerIOUsageEvent event = new ContainerIOUsageEvent(); + + event.serviceRequests = containerMetrics.getBlkIOServiceCount(); + event.dataTransferred = containerMetrics.getBlkIOServiced(); + event.commit(); + } + } + + private static void emitContainerCPUThrottling() { + if (containerMetrics != null) { + ContainerCPUThrottlingEvent event = new ContainerCPUThrottlingEvent(); + + event.cpuElapsedSlices = containerMetrics.getCpuNumPeriods(); + event.cpuThrottledSlices = containerMetrics.getCpuNumThrottled(); + event.cpuThrottledTime = containerMetrics.getCpuThrottledTime(); + event.commit(); + } + } + @SuppressWarnings("deprecation") public static byte[] retransformCallback(Class klass, byte[] oldBytes) throws Throwable { if (java.lang.Throwable.class == klass) { @@ -172,8 +261,14 @@ public final class JDKEvents { } public static void remove() { - RequestEngine.removeHook(JDKEvents::emitExceptionStatistics); + RequestEngine.removeHook(emitExceptionStatistics); RequestEngine.removeHook(emitDirectBufferStatistics); + + RequestEngine.removeHook(emitContainerConfiguration); + RequestEngine.removeHook(emitContainerCPUUsage); + RequestEngine.removeHook(emitContainerCPUThrottling); + RequestEngine.removeHook(emitContainerMemoryUsage); + RequestEngine.removeHook(emitContainerIOUsage); } private static void emitDirectBufferStatistics() { diff --git a/src/jdk.jfr/share/conf/jfr/default.jfc b/src/jdk.jfr/share/conf/jfr/default.jfc index b43e4858c93..13304286d72 100644 --- a/src/jdk.jfr/share/conf/jfr/default.jfc +++ b/src/jdk.jfr/share/conf/jfr/default.jfc @@ -554,6 +554,31 @@ beginChunk + + true + beginChunk + + + + true + 30 s + + + + true + 30 s + + + + true + 30 s + + + + true + 30 s + + true beginChunk diff --git a/src/jdk.jfr/share/conf/jfr/profile.jfc b/src/jdk.jfr/share/conf/jfr/profile.jfc index d8c7419264a..ebbffb7de13 100644 --- a/src/jdk.jfr/share/conf/jfr/profile.jfc +++ b/src/jdk.jfr/share/conf/jfr/profile.jfc @@ -554,6 +554,31 @@ beginChunk + + true + beginChunk + + + + true + 30 s + + + + true + 30 s + + + + true + 30 s + + + + true + 30 s + + true beginChunk diff --git a/test/hotspot/jtreg/containers/docker/JfrReporter.java b/test/hotspot/jtreg/containers/docker/JfrReporter.java index 0c3537f1073..d660415cecb 100644 --- a/test/hotspot/jtreg/containers/docker/JfrReporter.java +++ b/test/hotspot/jtreg/containers/docker/JfrReporter.java @@ -25,6 +25,7 @@ import java.nio.file.Path; import java.nio.file.Paths; import jdk.jfr.Recording; +import jdk.jfr.EventSettings; import jdk.jfr.ValueDescriptor; import jdk.jfr.consumer.RecordedEvent; import jdk.jfr.consumer.RecordingFile; @@ -34,7 +35,11 @@ public class JfrReporter { public static void main(String[] args) throws Exception { String eventName = args[0]; try(Recording r = new Recording()) { - r.enable(eventName); + EventSettings es = r.enable(eventName); + for (int i = 1; i < args.length; i++) { + String[] kv = args[i].split("="); + es = es.with(kv[0], kv[1]); + } r.start(); r.stop(); Path p = Paths.get("/", "tmp", eventName + ".jfr"); diff --git a/test/hotspot/jtreg/containers/docker/TestJFREvents.java b/test/hotspot/jtreg/containers/docker/TestJFREvents.java index 8bb2fb8a9d5..8e252d16d25 100644 --- a/test/hotspot/jtreg/containers/docker/TestJFREvents.java +++ b/test/hotspot/jtreg/containers/docker/TestJFREvents.java @@ -70,31 +70,114 @@ public class TestJFREvents { testProcessInfo(); testEnvironmentVariables(); + + containerInfoTestCase(); + testCpuUsage(); + testCpuThrottling(); + testMemoryUsage(); + testIOUsage(); } finally { DockerTestUtils.removeDockerImage(imageName); } } - // This test case is currently not in use. - // Once new Container events are available, this test case can be used to test - // processor-related configuration such as active processor count (see JDK-8203359). - private static void cpuTestCase() throws Exception { + private static void containerInfoTestCase() throws Exception { // leave one CPU for system and tools, otherwise this test may be unstable int maxNrOfAvailableCpus = availableCPUs - 1; for (int i=1; i < maxNrOfAvailableCpus; i = i * 2) { - testCPUInfo("jdk.ContainerConfiguration", i, i); + for (int j=64; j <= 256; j *= 2) { + testContainerInfo(i, j); + } } } - private static void testCPUInfo(String eventName, int valueToSet, int expectedValue) throws Exception { - Common.logNewTestCase("CPUInfo: --cpus = " + valueToSet); - String fieldName = "activeProcessorCount"; + private static void testContainerInfo(int expectedCPUs, int expectedMemoryMB) throws Exception { + Common.logNewTestCase("ContainerInfo: --cpus = " + expectedCPUs + " --memory=" + expectedMemoryMB + "m"); + String eventName = "jdk.ContainerConfiguration"; + long expectedSlicePeriod = 100000; // default slice period + long expectedMemoryLimit = expectedMemoryMB * 1024 * 1024; + + String cpuCountFld = "effectiveCpuCount"; + String cpuQuotaFld = "cpuQuota"; + String cpuSlicePeriodFld = "cpuSlicePeriod"; + String memoryLimitFld = "memoryLimit"; + DockerTestUtils.dockerRunJava( commonDockerOpts() - .addDockerOpts("--cpus=" + valueToSet) + .addDockerOpts("--cpus=" + expectedCPUs) + .addDockerOpts("--memory=" + expectedMemoryMB + "m") .addClassOptions(eventName)) .shouldHaveExitValue(0) - .shouldContain(fieldName + " = " + expectedValue); + .shouldContain(cpuCountFld + " = " + expectedCPUs) + .shouldContain(cpuSlicePeriodFld + " = " + expectedSlicePeriod) + .shouldContain(cpuQuotaFld + " = " + expectedCPUs * expectedSlicePeriod) + .shouldContain(memoryLimitFld + " = " + expectedMemoryLimit); + } + + private static void testCpuUsage() throws Exception { + Common.logNewTestCase("CPU Usage"); + String eventName = "jdk.ContainerCPUUsage"; + + String cpuTimeFld = "cpuTime"; + String cpuUserTimeFld = "cpuUserTime"; + String cpuSystemTimeFld = "cpuSystemTime"; + + DockerTestUtils.dockerRunJava( + commonDockerOpts() + .addClassOptions(eventName, "period=endChunk")) + .shouldHaveExitValue(0) + .shouldNotContain(cpuTimeFld + " = " + 0) + .shouldNotContain(cpuUserTimeFld + " = " + 0) + .shouldNotContain(cpuSystemTimeFld + " = " + 0); + } + + private static void testMemoryUsage() throws Exception { + Common.logNewTestCase("Memory Usage"); + String eventName = "jdk.ContainerMemoryUsage"; + + String memoryFailCountFld = "memoryFailCount"; + String memoryUsageFld = "memoryUsage"; + String swapMemoryUsageFld = "swapMemoryUsage"; + + DockerTestUtils.dockerRunJava( + commonDockerOpts() + .addClassOptions(eventName, "period=endChunk")) + .shouldHaveExitValue(0) + .shouldContain(memoryFailCountFld) + .shouldContain(memoryUsageFld) + .shouldContain(swapMemoryUsageFld); + } + + private static void testIOUsage() throws Exception { + Common.logNewTestCase("I/O Usage"); + String eventName = "jdk.ContainerIOUsage"; + + String serviceRequestsFld = "serviceRequests"; + String dataTransferredFld = "dataTransferred"; + + DockerTestUtils.dockerRunJava( + commonDockerOpts() + .addClassOptions(eventName, "period=endChunk")) + .shouldHaveExitValue(0) + .shouldContain(serviceRequestsFld) + .shouldContain(dataTransferredFld); + } + + private static void testCpuThrottling() throws Exception { + Common.logNewTestCase("CPU Throttling"); + String eventName = "jdk.ContainerCPUThrottling"; + + String cpuElapsedSlicesFld = "cpuElapsedSlices"; + String cpuThrottledSlicesFld = "cpuThrottledSlices"; + String cpuThrottledTimeFld = "cpuThrottledTime"; + + DockerTestUtils.dockerRunJava( + commonDockerOpts() + .addClassOptions(eventName, "period=endChunk")) + .shouldHaveExitValue(0) + .shouldContain(cpuElapsedSlicesFld) + .shouldContain(cpuThrottledSlicesFld) + .shouldContain(cpuThrottledTimeFld); } @@ -118,7 +201,6 @@ public class TestJFREvents { .shouldContain("pid = 1"); } - private static DockerRunOptions commonDockerOpts() { return new DockerRunOptions(imageName, "/jdk/bin/java", "JfrReporter") .addDockerOpts("--volume", Utils.TEST_CLASSES + ":/test-classes/") diff --git a/test/jdk/jdk/jfr/event/metadata/TestLookForUntestedEvents.java b/test/jdk/jdk/jfr/event/metadata/TestLookForUntestedEvents.java index 1f09bbb26ea..4100927f94e 100644 --- a/test/jdk/jdk/jfr/event/metadata/TestLookForUntestedEvents.java +++ b/test/jdk/jdk/jfr/event/metadata/TestLookForUntestedEvents.java @@ -71,6 +71,13 @@ public class TestLookForUntestedEvents { "GCPhasePauseLevel4") ); + // Container events are tested in hotspot/jtreg/containers/docker/TestJFREvents.java + private static final Set coveredContainerEvents = new HashSet<>( + Arrays.asList( + "ContainerConfiguration", "ContainerCPUUsage", "ContainerCPUThrottling", + "ContainerMemoryUsage", "ContainerIOUsage") + ); + // This is a "known failure list" for this test. // NOTE: if the event is not covered, a bug should be open, and bug number // noted in the comments for this set. @@ -115,6 +122,7 @@ public class TestLookForUntestedEvents { // Account for hard-to-test, experimental and GC tested events eventsNotCoveredByTest.removeAll(hardToTestEvents); eventsNotCoveredByTest.removeAll(coveredGcEvents); + eventsNotCoveredByTest.removeAll(coveredContainerEvents); eventsNotCoveredByTest.removeAll(knownNotCoveredEvents); if (!eventsNotCoveredByTest.isEmpty()) { diff --git a/test/lib/jdk/test/lib/jfr/EventNames.java b/test/lib/jdk/test/lib/jfr/EventNames.java index b4fd3652551..8de14e5eef3 100644 --- a/test/lib/jdk/test/lib/jfr/EventNames.java +++ b/test/lib/jdk/test/lib/jfr/EventNames.java @@ -201,6 +201,12 @@ public class EventNames { public final static String DirectBufferStatistics = PREFIX + "DirectBufferStatistics"; public final static String Deserialization = PREFIX + "Deserialization"; + // Containers + public static final String ContainerConfiguration = PREFIX + "ContainerConfiguration"; + public static final String ContainerCPUUsage = PREFIX + "ContainerCPUUsage"; + public static final String ContainerCPUThrottling = PREFIX + "ContainerCPUThrottling"; + public static final String ContainerMemoryUsage = PREFIX + "ContainerMemoryUsage"; + public static final String ContainerIOUsage = PREFIX + "ContainerIOUsage"; // Flight Recorder public final static String DumpReason = PREFIX + "DumpReason";