view test/jtreg/com/sun/javatest/lib/JavaCompileCommand.java @ 866:f42fc832db86

Add missing (ignored) lib/ files. Add test report and classes dirs to ignore. 2008-05-19 Mark Wielaard <mark@klomp.org> * test/jtreg/com/sun/javatest/lib/*.java: Added. * .hgignore: Add test/hotspot, test/jdk, test/langtools and test/jtreg/classes
author Mark Wielaard <mark@klomp.org>
date Mon, 19 May 2008 10:33:03 +0200
parents
children
line wrap: on
line source

/*
 * $Id$
 *
 * Copyright 1996-2008 Sun Microsystems, Inc.  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.  Sun designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
 * CA 95054 USA or visit www.sun.com if you need additional information or
 * have any questions.
 */
package com.sun.javatest.lib;

import java.io.OutputStream;
import java.io.PrintWriter;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.lang.reflect.InvocationTargetException;
import com.sun.javatest.Command;
import com.sun.javatest.Status;
import com.sun.javatest.util.PathClassLoader;
import com.sun.javatest.util.WriterStream;

/**
 * Invoke a Java compiler via reflection.
 * The compiler is assumed to have a constructor and compile method
 * matching the following signature:
 *<pre>
 * public class COMPILER {
 *     public COMPILER(java.io.OutputStream out, String compilerName);
 *     public boolean compile(String[] args);
 * }
 *</pre>
 * or
 *<pre>
 * public class COMPILER {
 *     public COMPILER();
 *     public int compile(String[] args);
 * }
 *</pre>
 * or
 * <pre>
 * public class COMPILER {
 *    public static int compile(String[] args, PrintWriter out);
 * }
 * </pre>
 * This means the command is suitable for (but not limited to) the
 * compiler javac suplied with JDK. (Note that this uses an internal
 * API of javac which is not documented and is not guaranteed to exist
 * in any specific release of JDK.)
 */
public class JavaCompileCommand extends Command
{
    /**
     * A stand-alone entry point for this command. An instance of this
     * command is created, and its <code>run</code> method invoked,
     * passing in the command line args and <code>System.out</code> and
     * <code>System.err</code> as the two streams.
     * @param args command line arguments for this command.
     * @see #run
     */
    public static void main(String[] args) {
	PrintWriter out = new PrintWriter(System.out);
	PrintWriter err = new PrintWriter(System.err);
	Status s;
	try {
	    JavaCompileCommand c = new JavaCompileCommand();
	    s = c.run(args, out, err);
	}
	finally {
	    out.flush();
	    err.flush();
	}
	s.exit();
    }


    /**
     * Invoke a specified compiler, or the default, javac.
     * If the first word in the <code>args</code> array is "-compiler"
     * the second is interpreted as the class name for the compiler to be 
     * invoked, optionally preceded by a name for the compiler, separated
     * from the class name by a colon.  If no -compiler is specified,
     * the default is `javac:com.sun.tools.javac.Main'. If -compiler is specified
     * but no compiler name is given before the class name, the default name
     * will be `java ' followed by the classname. For example, `-compiler Main'
     * will result in the class name being `Main' and the compiler name being
     * `java Main'. After determining the class and compiler name,
     * an instance of the compiler class will be created, passing it a stream
     * using the <code>ref</code> parameter, and the name of the compiler.
     * Then the `compile' method will be invoked, passing it the remaining
     * values of the `args' parameter.  If the compile method returns true,
     * the result will be a status of `passed'; if it returns `false', the
     * result will be `failed'. If any problems arise, the result will be
     * a status of `error'.
     * @param args An optional specification for the compiler to be invoked,
     *		followed by arguments for the compiler's compile method.
     * @param log  Not used.
     * @param ref  Passed to the compiler that is invoked.
     * @return `passed' if the compilation is successful; `failed' if the
     * 		compiler is invoked and errors are found in the file(s)
     *		being compiler; or `error' if some more serios problem arose
     *		that prevented the compiler performing its task.
     */
    public Status run(String[] args, PrintWriter log, PrintWriter ref) {

	if (args.length == 0)
	    return Status.error("No args supplied");

	String compilerClassName = null;
	String compilerName = null;
	String classpath = null;
	String[] options = null;

	// If we find a '-' in the args, what comes before it are 
	// options for this class and what comes after it are args 
	// for the compiler class. If don't find a '-', there are no
	// options for this class, and everything is handed off to
	// the compiler class

	for (int i = 0; i < args.length; i++) {
	    if (args[i].equals("-")) {
		options = new String[i];
		System.arraycopy(args, 0, options, 0, options.length);
		args = shift(args, i+1);
		break;
	    }
	}
	
	if (options != null) {
	    for (int i = 0; i < options.length; i++) {
		if (options[i].equals("-compiler")) {
		    if (i + 1 == options.length)
			return Status.error("No compiler specified after -compiler option");
		    
		    String s = options[++i];
		    int colon = s.indexOf(":");
		    if (colon == -1) {
			compilerClassName = s;
			compilerName = "java " + s;
		    }
		    else {
			compilerClassName = s.substring(colon + 1);
			compilerName = s.substring(0, colon);
		    }
		}
		else if (options[i].equals("-cp") || options[i].equals("-classpath")) {
		    if (i + 1 == options.length)
			return Status.error("No path specified after -cp or -classpath option");
		    classpath = options[++i];
		}
		else if (options[i].equals("-verbose"))
		    verbose = true;
		else
		    return Status.error("Unrecognized option: " + options[i]);
	    }
	}

	this.log = log;

	try {

	    ClassLoader loader;
	    if (classpath == null)
		loader = null;
	    else
		loader = new PathClassLoader(classpath);

	    Class compilerClass;
	    if (compilerClassName != null) {
		compilerClass = getClass(loader, compilerClassName);
		if (compilerClass == null)
		    return Status.error("Cannot find compiler: " + compilerClassName);
	    }
	    else {
		compilerName = "javac";
		compilerClass = getClass(loader, "com.sun.tools.javac.Main");  // JDK1.3+
		if (compilerClass == null)
		    compilerClass = getClass(loader, "sun.tools.javac.Main");  // JDK1.1-2
		if (compilerClass == null)
		    return Status.error("Cannot find compiler");
	    }

	    loader = null;

	    Object[] compileMethodArgs;
	    Method compileMethod = getMethod(compilerClass, "compile", // JDK1.4+
					     new Class[] { String[].class, PrintWriter.class });
	    if (compileMethod != null) 
		compileMethodArgs = new Object[] { args, ref };
	    else {
		compileMethod = getMethod(compilerClass, "compile",   // JDK1.1-3
					  new Class[] { String[].class });
		if (compileMethod != null)
		    compileMethodArgs = new Object[] { args };
		else
		    return Status.error("Cannot find compile method for " + compilerClass.getName());
	    }

	    Object compiler;
	    if (Modifier.isStatic(compileMethod.getModifiers()))
		compiler =  null;
	    else {
		Object[] constrArgs;
		Constructor constr = getConstructor(compilerClass, // JDK1.1-2
						    new Class[] { OutputStream.class, String.class });
		if (constr != null) 
		    constrArgs = new Object[] { new WriterStream(ref), compilerName };
		else {
		    constr = getConstructor(compilerClass, new Class[0]); // JDK1.3
		    if (constr != null)
			constrArgs = new Object[0];
		    else
			return Status.error("Cannot find suitable constructor for " + compilerClass.getName());
		}
		try {
		    compiler = constr.newInstance(constrArgs);
		}
		catch (Throwable t) {
		    t.printStackTrace(log);
		    return Status.error("Cannot instantiate compiler");
		}
	    }

	    Object result;
	    try {
		result = compileMethod.invoke(compiler, compileMethodArgs);
	    }
	    catch (Throwable t) {
		t.printStackTrace(log);
		return Status.error("Error invoking compiler");
	    }

	    // result might be a boolean (old javac) or an int (new javac)
	    if (result instanceof Boolean) {
		boolean ok = ((Boolean)result).booleanValue();
		return (ok ? passed : failed);
	    }
	    else if (result instanceof Integer) {
		int rc = ((Integer)result).intValue();
		return (rc == 0 ? passed : failed);
	    }
	    else 
		return Status.error("Unexpected return value from compiler: " + result);
	}
	finally {
	    log.flush();
	    ref.flush();
	}    
    }

    private Class getClass(ClassLoader loader, String name) {
	try { 
	    return (loader == null ? Class.forName(name) : loader.loadClass(name));
	}
	catch (ClassNotFoundException e) {
	    return null;
	}
    }

    private Constructor getConstructor(Class c, Class[] argTypes) {
	try {
	    return c.getConstructor(argTypes);
	}
	catch (NoSuchMethodException e) {
	    return null;
	}
	catch (Throwable t) {
	    if (verbose)
		t.printStackTrace(log);
	    return null;
	}
    }

    private Method getMethod(Class c, String name, Class[] argTypes) {
	try {
	    return c.getMethod(name, argTypes);
	}
	catch (NoSuchMethodException e) {
	    return null;
	}
	catch (Throwable t) {
	    if (verbose)
		t.printStackTrace(log);
	    return null;
	}
    }

    private static String[] shift(String[] args, int n) {
	String[] newArgs = new String[args.length - n];
	System.arraycopy(args, n, newArgs, 0, newArgs.length);
	return newArgs;
    }

    public static boolean defaultVerbose = Boolean.getBoolean("javatest.JavaCompileCommand.verbose");
    private boolean verbose = defaultVerbose;
    private PrintWriter log;

    private static final Status passed = Status.passed("Compilation successful");
    private static final Status failed = Status.failed("Compilation failed");
}