/* * Copyright (c) 2017, 2022, Oracle and/or its affiliates. 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. * * 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. */ /* * @test * @key cgroups * @summary Test JVM's CPU resource awareness when running inside docker container * @requires docker.support * @library /test/lib * @modules java.base/jdk.internal.misc * java.base/jdk.internal.platform * java.management * jdk.jartool/sun.tools.jar * @build PrintContainerInfo CheckOperatingSystemMXBean * @run driver TestCPUAwareness */ import java.util.List; import jdk.test.lib.process.OutputAnalyzer; import jdk.test.lib.containers.docker.Common; import jdk.test.lib.containers.docker.DockerRunOptions; import jdk.test.lib.containers.docker.DockerTestUtils; import jdk.test.lib.containers.cgroup.CPUSetsReader; public class TestCPUAwareness { private static final String imageName = Common.imageName("cpu"); private static final int availableCPUs = Runtime.getRuntime().availableProcessors(); public static void main(String[] args) throws Exception { if (!DockerTestUtils.canTestDocker()) { return; } System.out.println("Test Environment: detected availableCPUs = " + availableCPUs); DockerTestUtils.buildJdkContainerImage(imageName); try { // cpuset, period, shares, expected Active Processor Count testComboWithCpuSets(); // cpu shares - it should be safe to use CPU shares exceeding available CPUs testCpuShares(256, 1); testCpuShares(2048, 2); testCpuShares(4096, 4); // 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) { testCpus(i, i); } // If ActiveProcessorCount is set, the VM should use it, regardless of other // container settings, host settings or available CPUs on the host. testActiveProcessorCount(1, 1); testActiveProcessorCount(2, 2); // cpu quota and period testCpuQuotaAndPeriod(50*1000, 100*1000); testCpuQuotaAndPeriod(100*1000, 100*1000); testCpuQuotaAndPeriod(150*1000, 100*1000); testCpuQuotaAndPeriod(400*1000, 100*1000); testOperatingSystemMXBeanAwareness("0.5", "1"); testOperatingSystemMXBeanAwareness("1.0", "1"); if (availableCPUs > 2) { testOperatingSystemMXBeanAwareness("1.2", "2"); testOperatingSystemMXBeanAwareness("1.8", "2"); testOperatingSystemMXBeanAwareness("2.0", "2"); } } finally { if (!DockerTestUtils.RETAIN_IMAGE_AFTER_TEST) { DockerTestUtils.removeDockerImage(imageName); } } } private static void testComboWithCpuSets() throws Exception { String cpuSetStr = CPUSetsReader.readFromProcStatus("Cpus_allowed_list"); System.out.println("cpuSetStr = " + cpuSetStr); // OLD = use the deprecated -XX:+UseContainerCpuShares flag, which // will be removed in the next JDK release. See JDK-8281181. boolean OLD = true; boolean NEW = false; if (cpuSetStr == null) { System.out.printf("The cpuset test cases are skipped"); } else { List cpuSet = CPUSetsReader.parseCpuSet(cpuSetStr); // Test subset of cpuset with one element if (cpuSet.size() >= 1) { String testCpuSet = CPUSetsReader.listToString(cpuSet, 1); testAPCCombo(OLD, testCpuSet, 200*1000, 100*1000, 4*1024, true, 1); testAPCCombo(NEW, testCpuSet, 200*1000, 100*1000, 4*1024, true, 1); } // Test subset of cpuset with two elements if (cpuSet.size() >= 2) { String testCpuSet = CPUSetsReader.listToString(cpuSet, 2); testAPCCombo(OLD, testCpuSet, 200*1000, 100*1000, 4*1024, true, 2); testAPCCombo(OLD, testCpuSet, 200*1000, 100*1000, 1023, true, 2); testAPCCombo(OLD, testCpuSet, 200*1000, 100*1000, 1023, false,1); testAPCCombo(NEW, testCpuSet, 200*1000, 100*1000, 4*1024, true, 2); testAPCCombo(NEW, testCpuSet, 200*1000, 100*1000, 1023, true, 2); testAPCCombo(NEW, testCpuSet, 200*1000, 100*1000, 1023, false,2); } // Test subset of cpuset with three elements if (cpuSet.size() >= 3) { String testCpuSet = CPUSetsReader.listToString(cpuSet, 3); testAPCCombo(OLD, testCpuSet, 100*1000, 100*1000, 2*1024, true, 1); testAPCCombo(OLD, testCpuSet, 200*1000, 100*1000, 1023, true, 2); testAPCCombo(OLD, testCpuSet, 200*1000, 100*1000, 1023, false,1); testAPCCombo(NEW, testCpuSet, 100*1000, 100*1000, 2*1024, true, 1); testAPCCombo(NEW, testCpuSet, 200*1000, 100*1000, 1023, true, 2); testAPCCombo(NEW, testCpuSet, 200*1000, 100*1000, 1023, false,2); } } } private static void testActiveProcessorCount(int valueToSet, int expectedValue) throws Exception { Common.logNewTestCase("Test ActiveProcessorCount: valueToSet = " + valueToSet); DockerRunOptions opts = Common.newOpts(imageName) .addJavaOpts("-XX:ActiveProcessorCount=" + valueToSet, "-Xlog:os=trace"); Common.run(opts) .shouldMatch("active processor count set by user.*" + expectedValue); } private static void testCpus(int valueToSet, int expectedTraceValue) throws Exception { Common.logNewTestCase("test cpus: " + valueToSet); DockerRunOptions opts = Common.newOpts(imageName) .addDockerOpts("--cpu-period=" + 10000) .addDockerOpts("--cpu-quota=" + valueToSet * 10000); Common.run(opts) .shouldMatch("active_processor_count.*" + expectedTraceValue); } // Expected active processor count can not exceed available CPU count private static int adjustExpectedAPCForAvailableCPUs(int expectedAPC) { if (expectedAPC > availableCPUs) { expectedAPC = availableCPUs; System.out.println("Adjusted expectedAPC = " + expectedAPC); } return expectedAPC; } private static void testCpuQuotaAndPeriod(int quota, int period) throws Exception { Common.logNewTestCase("test cpu quota and period: "); System.out.println("quota = " + quota); System.out.println("period = " + period); int expectedAPC = (int) Math.ceil((float) quota / (float) period); System.out.println("expectedAPC = " + expectedAPC); expectedAPC = adjustExpectedAPCForAvailableCPUs(expectedAPC); DockerRunOptions opts = Common.newOpts(imageName) .addDockerOpts("--cpu-period=" + period) .addDockerOpts("--cpu-quota=" + quota); Common.run(opts) .shouldMatch("CPU Period is.*" + period) .shouldMatch("CPU Quota is.*" + quota) .shouldMatch("active_processor_count.*" + expectedAPC); } // Test correctess of automatically selected active processor count // Note: when -XX:+UseContainerCpuShares is removed, // useContainerCpuShares, shares, and usePreferContainerQuotaForCPUCount // should also be removed. private static void testAPCCombo(boolean useContainerCpuShares, String cpuset, int quota, int period, int shares, boolean usePreferContainerQuotaForCPUCount, int expectedAPC) throws Exception { Common.logNewTestCase("test APC Combo"); System.out.println("cpuset = " + cpuset); System.out.println("quota = " + quota); System.out.println("period = " + period); System.out.println("shares = " + shares); System.out.println("useContainerCpuShares = " + useContainerCpuShares); System.out.println("usePreferContainerQuotaForCPUCount = " + usePreferContainerQuotaForCPUCount); System.out.println("expectedAPC = " + expectedAPC); expectedAPC = adjustExpectedAPCForAvailableCPUs(expectedAPC); DockerRunOptions opts = Common.newOpts(imageName) .addDockerOpts("--cpuset-cpus", "" + cpuset) .addDockerOpts("--cpu-period=" + period) .addDockerOpts("--cpu-quota=" + quota) .addDockerOpts("--cpu-shares=" + shares); if (useContainerCpuShares) opts.addJavaOpts("-XX:+UseContainerCpuShares"); // deprecated if (!usePreferContainerQuotaForCPUCount) opts.addJavaOpts("-XX:-PreferContainerQuotaForCPUCount"); // deprecated Common.run(opts) .shouldMatch("active_processor_count.*" + expectedAPC); } // Note: when -XX:+UseContainerCpuShares is removed, this test should also be removed. private static void testCpuShares(int shares, int expectedAPC) throws Exception { Common.logNewTestCase("test cpu shares, shares = " + shares); System.out.println("expectedAPC = " + expectedAPC); expectedAPC = adjustExpectedAPCForAvailableCPUs(expectedAPC); DockerRunOptions opts = Common.newOpts(imageName) .addDockerOpts("--cpu-shares=" + shares); opts.addJavaOpts("-XX:+UseContainerCpuShares"); // deprecated OutputAnalyzer out = Common.run(opts); // Cgroups v2 needs to do some scaling of raw shares values. Hence, // 256 CPU shares come back as 264. Raw value written to cpu.weight // is 10. The reason this works for >= 1024 shares value is because // post-scaling the closest multiple of 1024 is found and returned. // // For values < 1024, this doesn't happen so loosen the match to a // 3-digit number and ensure the active_processor_count is as // expected. if (shares < 1024) { out.shouldMatch("CPU Shares is.*\\d{3}"); } else { out.shouldMatch("CPU Shares is.*" + shares); } out.shouldMatch("active_processor_count.*" + expectedAPC); } private static void testOperatingSystemMXBeanAwareness(String cpuAllocation, String expectedCpus) throws Exception { Common.logNewTestCase("Check OperatingSystemMXBean"); DockerRunOptions opts = Common.newOpts(imageName, "CheckOperatingSystemMXBean") .addDockerOpts( "--cpus", cpuAllocation ) // CheckOperatingSystemMXBean uses Metrics (jdk.internal.platform) for // diagnostics .addJavaOpts("--add-exports") .addJavaOpts("java.base/jdk.internal.platform=ALL-UNNAMED"); DockerTestUtils.dockerRunJava(opts) .shouldHaveExitValue(0) .shouldContain("Checking OperatingSystemMXBean") .shouldContain("Runtime.availableProcessors: " + expectedCpus) .shouldContain("OperatingSystemMXBean.getAvailableProcessors: " + expectedCpus) .shouldMatch("OperatingSystemMXBean\\.getSystemCpuLoad: [0-9]+\\.[0-9]+") .shouldMatch("OperatingSystemMXBean\\.getCpuLoad: [0-9]+\\.[0-9]+") ; } }