/*
 * Copyright (c) 1999, 2003, 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 sun.awt;

import java.io.*;

import java.util.*;
import java.util.logging.*;

/*
 * Internal class that manages sun.awt.Debug settings.
 * Settings can be specified on a global, per-package,
 * or per-class level.
 *
 * Properties affecting the behaviour of the Debug class are
 * loaded from the awtdebug.properties file at class load
 * time. The properties file is assumed to be in the
 * user.home directory. A different file can be used
 * by setting the awtdebug.properties system property.
 *      e.g. java -Dawtdebug.properties=foo.properties
 *
 * Only properties beginning with 'awtdebug' have any
 * meaning-- all other properties are ignored.
 *
 * You can override the properties file by specifying
 * 'awtdebug' props as system properties on the command line.
 *      e.g. java -Dawtdebug.trace=true
 * Properties specific to a package or a class can be set
 * by qualifying the property names as follows:
 *      awtdebug.<property name>.<class or package name>
 * So for example, turning on tracing in the com.acme.Fubar
 * class would be done as follows:
 *      awtdebug.trace.com.acme.Fubar=true
 *
 * Class settings always override package settings, which in
 * turn override global settings.
 *
 * Addition from July, 2007.
 *
 * After the fix for 4638447 all the usage of DebugHelper
 * classes in Java code are replaced with the corresponding
 * Java Logging API calls. This file is now used only to
 * control native logging.
 *
 * To enable native logging you should set the following
 * system property to 'true': sun.awt.nativedebug. After
 * the native logging is enabled, the actual debug settings
 * are read the same way as described above (as before
 * the fix for 4638447).
 */
final class DebugSettings {
    private static final Logger log = Logger.getLogger("sun.awt.debug.DebugSettings");

    /* standard debug property key names */
    static final String PREFIX = "awtdebug";
    static final String PROP_FILE = "properties";

    /* default property settings */
    private static final String DEFAULT_PROPS[] = {
        "awtdebug.assert=true",
        "awtdebug.trace=false",
        "awtdebug.on=true",
        "awtdebug.ctrace=false"
    };

    /* global instance of the settings object */
    private static DebugSettings        instance = null;

    private Properties  props = new Properties();

    static void init() {
        if (instance != null) {
            return;
        }

        NativeLibLoader.loadLibraries();
        instance = new DebugSettings();
        instance.loadNativeSettings();
    }

    private DebugSettings() {
        new java.security.PrivilegedAction() {
            public Object run() {
                loadProperties();
                return null;
            }
        }.run();
    }

    /*
     * Load debug properties from file, then override
     * with any command line specified properties
     */
    private synchronized void loadProperties() {
        // setup initial properties
        java.security.AccessController.doPrivileged(
            new java.security.PrivilegedAction()
        {
            public Object run() {
                loadDefaultProperties();
                loadFileProperties();
                loadSystemProperties();
                return null;
            }
        });

        // echo the initial property settings to stdout
        if (log.isLoggable(Level.FINE)) {
            log.log(Level.FINE, "DebugSettings:\n{0}", String.valueOf(this));
        }
    }

    public String toString() {
        Enumeration enum_ = props.propertyNames();
        ByteArrayOutputStream bout = new ByteArrayOutputStream();
        PrintStream pout = new PrintStream(bout);

        while (enum_.hasMoreElements()) {
            String key = (String)enum_.nextElement();
            String value = props.getProperty(key, "");
            pout.println(key + " = " + value);
        }
        return new String(bout.toByteArray());
    }

    /*
     * Sets up default property values
     */
    private void loadDefaultProperties() {
        // is there a more inefficient way to setup default properties?
        // maybe, but this has got to be close to 100% non-optimal
        try {
            for ( int nprop = 0; nprop < DEFAULT_PROPS.length; nprop++ ) {
                StringBufferInputStream in = new StringBufferInputStream(DEFAULT_PROPS[nprop]);
                props.load(in);
                in.close();
            }
        } catch(IOException ioe) {
        }
    }

    /*
     * load properties from file, overriding defaults
     */
    private void loadFileProperties() {
        String          propPath;
        Properties      fileProps;

        // check if the user specified a particular settings file
        propPath = System.getProperty(PREFIX + "." + PROP_FILE, "");
        if (propPath.equals("")) {
        // otherwise get it from the user's home directory
            propPath = System.getProperty("user.home", "") +
                        File.separator +
                        PREFIX + "." + PROP_FILE;
        }

        File    propFile = new File(propPath);
        try {
            println("Reading debug settings from '" + propFile.getCanonicalPath() + "'...");
            FileInputStream     fin = new FileInputStream(propFile);
            props.load(fin);
            fin.close();
        } catch ( FileNotFoundException fne ) {
            println("Did not find settings file.");
        } catch ( IOException ioe ) {
            println("Problem reading settings, IOException: " + ioe.getMessage());
        }
    }

    /*
     * load properties from system props (command line spec'd usually),
     * overriding default or file properties
     */
    private void loadSystemProperties() {
        // override file properties with system properties
        Properties sysProps = System.getProperties();
        Enumeration enum_ = sysProps.propertyNames();
        while ( enum_.hasMoreElements() ) {
            String key = (String)enum_.nextElement();
            String value = sysProps.getProperty(key,"");
            // copy any "awtdebug" properties over
            if ( key.startsWith(PREFIX) ) {
                props.setProperty(key, value);
            }
        }
    }

    
Gets named boolean property
Params:
  • key – Name of property
  • defval – Default value if property does not exist
Returns:boolean value of the named property
/** * Gets named boolean property * @param key Name of property * @param defval Default value if property does not exist * @return boolean value of the named property */
public synchronized boolean getBoolean(String key, boolean defval) { String value = getString(key, String.valueOf(defval)); return value.equalsIgnoreCase("true"); }
Gets named integer property
Params:
  • key – Name of property
  • defval – Default value if property does not exist
Returns:integer value of the named property
/** * Gets named integer property * @param key Name of property * @param defval Default value if property does not exist * @return integer value of the named property */
public synchronized int getInt(String key, int defval) { String value = getString(key, String.valueOf(defval)); return Integer.parseInt(value); }
Gets named String property
Params:
  • key – Name of property
  • defval – Default value if property does not exist
Returns:string value of the named property
/** * Gets named String property * @param key Name of property * @param defval Default value if property does not exist * @return string value of the named property */
public synchronized String getString(String key, String defval) { String actualKeyName = PREFIX + "." + key; String value = props.getProperty(actualKeyName, defval); //println(actualKeyName+"="+value); return value; } public synchronized Enumeration getPropertyNames() { Vector propNames = new Vector(); Enumeration enum_ = props.propertyNames(); // remove global prefix from property names while ( enum_.hasMoreElements() ) { String propName = (String)enum_.nextElement(); propName = propName.substring(PREFIX.length()+1); propNames.addElement(propName); } return propNames.elements(); } private void println(Object object) { if (log.isLoggable(Level.FINER)) { log.log(Level.FINER, object.toString()); } } private static final String PROP_CTRACE = "ctrace"; private static final int PROP_CTRACE_LEN = PROP_CTRACE.length(); private native synchronized void setCTracingOn(boolean enabled); private native synchronized void setCTracingOn(boolean enabled, String file); private native synchronized void setCTracingOn(boolean enabled, String file, int line); private void loadNativeSettings() { boolean ctracingOn; ctracingOn = getBoolean(PROP_CTRACE, false); setCTracingOn(ctracingOn); // // Filter out file/line ctrace properties from debug settings // Vector traces = new Vector(); Enumeration enum_ = getPropertyNames(); while ( enum_.hasMoreElements() ) { String key = (String)enum_.nextElement(); if ( key.startsWith(PROP_CTRACE) && key.length() > PROP_CTRACE_LEN ) { traces.addElement(key); } } // sort traces list so file-level traces will be before line-level ones Collections.sort(traces); // // Setup the trace points // Enumeration enumTraces = traces.elements(); while ( enumTraces.hasMoreElements() ) { String key = (String)enumTraces.nextElement(); String trace = key.substring(PROP_CTRACE_LEN+1); String filespec; String linespec; int delim= trace.indexOf('@'); boolean enabled; // parse out the filename and linenumber from the property name filespec = delim != -1 ? trace.substring(0, delim) : trace; linespec = delim != -1 ? trace.substring(delim+1) : ""; enabled = getBoolean(key, false); //System.out.println("Key="+key+", File="+filespec+", Line="+linespec+", Enabled="+enabled); if ( linespec.length() == 0 ) { // set file specific trace setting setCTracingOn(enabled, filespec); } else { // set line specific trace setting int linenum = Integer.parseInt(linespec, 10); setCTracingOn(enabled, filespec, linenum); } } } }