hotspot/test/testlibrary_tests/whitebox/vm_flags/VmFlagTest.java
author dcubed
Fri, 31 Oct 2014 10:15:29 -0700
changeset 27464 37671dcc4d8d
parent 27417 576e2b527e1c
child 30254 224dcf3ebade
permissions -rw-r--r--
Merge

/*
 * Copyright (c) 2014, 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.
 */

import java.util.Objects;
import java.util.function.BiConsumer;
import java.util.function.Function;
import sun.hotspot.WhiteBox;
import sun.management.*;
import com.sun.management.*;
import com.oracle.java.testlibrary.*;

public final class VmFlagTest<T> {
    public static final WhiteBox WHITE_BOX = WhiteBox.getWhiteBox();

    private static final String NONEXISTENT_FLAG = "NonexistentFlag";
    private final String flagName;
    private final BiConsumer<T, T> test;
    private final BiConsumer<String, T> set;
    private final Function<String, T> get;
    private final boolean isPositive;

    protected VmFlagTest(String flagName, BiConsumer<String, T> set,
            Function<String, T> get, boolean isPositive) {
        this.flagName = flagName;
        this.set = set;
        this.get = get;
        this.isPositive = isPositive;
        if (isPositive) {
            test = this::testWritePositive;
        } else {
            test = this::testWriteNegative;
        }
    }

    private void setNewValue(T value) {
        set.accept(flagName, value);
    }

    private T getValue() {
        return get.apply(flagName);
    }

    protected static <T> void runTest(String existentFlag, T[] tests,
            BiConsumer<String, T> set, Function<String, T> get) {
        runTest(existentFlag, tests, tests, set, get);
    }

    protected static <T> void runTest(String existentFlag, Function<String, T> get) {
        runTest(existentFlag, null, null, null, get);
    }

    protected static <T> void runTest(String existentFlag, T[] tests,
            T[] results, BiConsumer<String, T> set, Function<String, T> get) {
        if (existentFlag != null) {
            new VmFlagTest(existentFlag, set, get, true).test(tests, results);
        }
        new VmFlagTest(NONEXISTENT_FLAG, set, get, false).test(tests, results);
    }

    public final void test(T[] tests, T[] results) {
        if (isPositive) {
            testRead();
        }
        if (tests != null) {
            Asserts.assertEQ(tests.length, results.length, "[TESTBUG] tests.length != results.length");
            for (int i = 0, n = tests.length ; i < n; ++i) {
                test.accept(tests[i], results[i]);
            }
        }
    }

    protected String getVMOptionAsString() {
        if (WHITE_BOX.isConstantVMFlag(flagName) || WHITE_BOX.isLockedVMFlag(flagName)) {
          // JMM cannot access debug flags in product builds or locked flags,
          // use whitebox methods to get such flags value.
          return asString(getValue());
        }
        HotSpotDiagnosticMXBean diagnostic
                = ManagementFactoryHelper.getDiagnosticMXBean();
        VMOption tmp;
        try {
            tmp = diagnostic.getVMOption(flagName);
        } catch (IllegalArgumentException e) {
            tmp = null;
        }
        return tmp == null ? null : tmp.getValue();
    }

    private String testRead() {
        String value = getVMOptionAsString();
        Asserts.assertNotNull(value);
        Asserts.assertEQ(value, asString(getValue()));
        Asserts.assertEQ(value, asString(WHITE_BOX.getVMFlag(flagName)));
        return value;
    }

    private void testWritePositive(T value, T expected) {
        setNewValue(value);
        String newValue = testRead();
        Asserts.assertEQ(newValue, asString(expected));
    }

    private void testWriteNegative(T value, T expected) {
        // Should always return false for non-existing flags
        Asserts.assertFalse(WHITE_BOX.isConstantVMFlag(flagName));
        Asserts.assertFalse(WHITE_BOX.isLockedVMFlag(flagName));
        String oldValue = getVMOptionAsString();
        Asserts.assertEQ(oldValue, asString(getValue()));
        Asserts.assertEQ(oldValue, asString(WHITE_BOX.getVMFlag(flagName)));
        setNewValue(value);
        String newValue = getVMOptionAsString();
        Asserts.assertEQ(oldValue, newValue);
    }

    private String asString(Object value) {
        return value == null ? null : "" + value;
    }
}