src/jdk.jshell/share/classes/jdk/internal/jshell/debug/InternalDebugControl.java
author darcy
Wed, 25 Apr 2018 22:12:06 -0700
changeset 49895 661ef62a6618
parent 48373 f9152f462cbc
permissions -rw-r--r--
8200478: For boxing conversion javac uses Long.valueOf which does not guarantee caching according to its javadoc Reviewed-by: bpb

/*
 * Copyright (c) 2015, 2017, 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.  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.internal.jshell.debug;

import java.io.PrintStream;
import java.util.HashMap;
import java.util.Map;
import jdk.jshell.JShell;

/**
 * This class is used to externally control output messages for debugging the
 * implementation of the JShell API.
 * <p>
 * This is not part of the SPI nor API.
 */
public class InternalDebugControl {

    /**
     * This is a static only class; The constructor should never be called.
     */
    private InternalDebugControl() {
    }

    /**
     * General debugging.
     */
    public static final int DBG_GEN = 0b0000001;

    /**
     * File manager debuging.
     */
    public static final int DBG_FMGR = 0b0000010;

    /**
     * Completion analysis debugging.
     */
    public static final int DBG_COMPA = 0b0000100;

    /**
     * Dependency debugging.
     */
    public static final int DBG_DEP = 0b0001000;

    /**
     * Event debugging.
     */
    public static final int DBG_EVNT = 0b0010000;

    /**
     * Event debugging.
     */
    public static final int DBG_WRAP = 0b0100000;

    private static Map<JShell, Integer> debugMap = null;

    /**
     * Sets which debug flags are enabled for a given JShell instance. The flags
     * are or'ed bits as defined in {@code DBG_*}.
     *
     * @param state the JShell instance
     * @param flags the or'ed debug bits
     */
    public static void setDebugFlags(JShell state, int flags) {
        if (debugMap == null) {
            debugMap = new HashMap<>();
        }
        debugMap.put(state, flags);
    }

    /**
     * Release a JShell instance.
     *
     * @param state the JShell instance
     */
    public static void release(JShell state) {
        if (debugMap != null) {
            debugMap.remove(state);
        }
    }

    /**
     * Tests if any of the specified debug flags are enabled.
     *
     * @param state the JShell instance
     * @param flag the {@code DBG_*} bits to check
     * @return true if any of the flags are enabled
     */
    public static boolean isDebugEnabled(JShell state, int flag) {
        if (debugMap == null) {
            return false;
        }
        Integer flags = debugMap.get(state);
        if (flags == null) {
            return false;
        }
        return (flags & flag) != 0;
    }

    /**
     * Displays debug info if the specified debug flags are enabled.
     *
     * @param state the current JShell instance
     * @param err the {@code PrintStream} to report on
     * @param flags {@code DBG_*} flag bits to check
     * @param format format string for the output
     * @param args args for the format string
     */
    public static void debug(JShell state, PrintStream err, int flags, String format, Object... args) {
        if (isDebugEnabled(state, flags)) {
            err.printf(format, args);
        }
    }

    /**
     * Displays a fatal exception as debug info.
     *
     * @param state the current JShell instance
     * @param err the {@code PrintStream} to report on
     * @param ex the fatal Exception
     * @param where additional context
     */
    public static void debug(JShell state, PrintStream err, Throwable ex, String where) {
        if (isDebugEnabled(state, 0xFFFFFFFF)) {
            err.printf("Fatal error: %s: %s\n", where, ex.getMessage());
            ex.printStackTrace(err);
        }
    }
}