langtools/test/tools/javah/compareTest/CompareTest.java
author jwilhelm
Thu, 07 Feb 2013 15:51:25 +0100
changeset 15605 bf7de87bbe3a
parent 5520 86e4b9a9da40
permissions -rw-r--r--
8006432: Ratio flags should be unsigned Summary: Flags changed to be of uintx type Reviewed-by: johnc, tamao

/*
 * Copyright (c) 2009, 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.io.DataInputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Enumeration;
import java.util.List;
import java.util.Set;
import java.util.TreeSet;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;

import com.sun.tools.classfile.AccessFlags;
import com.sun.tools.classfile.ClassFile;
import com.sun.tools.classfile.ConstantPoolException;
import com.sun.tools.classfile.Method;
import java.io.BufferedReader;
import java.io.FileInputStream;
import java.io.InputStreamReader;
import java.util.LinkedHashSet;

public class CompareTest {
    String[][] testCases = {
        { },
        { "-jni" },
//        { "-llni" },
    };

    public static void main(String... args) throws Exception {
        new CompareTest().run(args);
    }

    public void run(String... args) throws Exception {
        old_javah_cmd = new File(args[0]);
        rt_jar = new File(args[1]);

        Set<String> testClasses;
        if (args.length > 2) {
            testClasses = new LinkedHashSet<String>(Arrays.asList(Arrays.copyOfRange(args, 2, args.length)));
        } else
            testClasses = getNativeClasses(new JarFile(rt_jar));

        for (String[] options: testCases) {
            for (String name: testClasses) {
                test(Arrays.asList(options), rt_jar, name);
            }
        }

        if (errors == 0)
            System.out.println(count + " tests passed");
        else
            throw new Exception(errors + "/" + count + " tests failed");
    }

    public void test(List<String> options, File bootclasspath, String className)
            throws IOException, InterruptedException {
        System.err.println("test: " + options + " " + className);
        count++;

        testOptions = options;
        testClassName = className;

        File oldOutDir = initDir(file(new File("old"), className));
        int old_rc = old_javah(options, oldOutDir, bootclasspath, className);

        File newOutDir = initDir(file(new File("new"), className));
        int new_rc = new_javah(options, newOutDir, bootclasspath, className);

        if (old_rc != new_rc)
            error("return codes differ; old: " + old_rc + ", new: " + new_rc);

        compare(oldOutDir, newOutDir);
    }

    int old_javah(List<String> options, File outDir, File bootclasspath, String className)
            throws IOException, InterruptedException {
        List<String> cmd = new ArrayList<String>();
        cmd.add(old_javah_cmd.getPath());
        cmd.addAll(options);
        cmd.add("-d");
        cmd.add(outDir.getPath());
        cmd.add("-bootclasspath");
        cmd.add(bootclasspath.getPath());
        cmd.add(className);
        System.err.println("old_javah: " + cmd);
        ProcessBuilder pb = new ProcessBuilder(cmd);
        pb.redirectErrorStream(true);
        Process p = pb.start();
        BufferedReader in = new BufferedReader(new InputStreamReader(p.getInputStream()));
        String line;
        StringBuilder sb = new StringBuilder();
        while ((line = in.readLine()) != null) {
            sb.append(line);
            sb.append("\n");
        }
        System.err.println("old javah out: " + sb.toString());
        return p.waitFor();
    }

    int new_javah(List<String> options, File outDir, File bootclasspath, String className) {
        List<String> args = new ArrayList<String>();
        args.addAll(options);
        args.add("-d");
        args.add(outDir.getPath());
        args.add("-bootclasspath");
        args.add(bootclasspath.getPath());
        args.add(className);
        StringWriter sw = new StringWriter();
        PrintWriter pw = new PrintWriter(sw);
        int rc = com.sun.tools.javah.Main.run(args.toArray(new String[args.size()]), pw);
        pw.close();
        System.err.println("new javah out: " + sw.toString());
        return rc;
    }

    Set<String> getNativeClasses(JarFile jar) throws IOException, ConstantPoolException {
        System.err.println("getNativeClasses: " + jar.getName());
        Set<String> results = new TreeSet<String>();
        Enumeration<JarEntry> e = jar.entries();
        while (e.hasMoreElements()) {
            JarEntry je = e.nextElement();
            if (isNativeClass(jar, je)) {
                String name = je.getName();
                results.add(name.substring(0, name.length() - 6).replace("/", "."));
            }
        }
        return results;
    }

    boolean isNativeClass(JarFile jar, JarEntry entry) throws IOException, ConstantPoolException {
        String name = entry.getName();
        if (name.startsWith("META-INF") || !name.endsWith(".class"))
            return false;
        //String className = name.substring(0, name.length() - 6).replace("/", ".");
        //System.err.println("check " + className);
        InputStream in = jar.getInputStream(entry);
        ClassFile cf = ClassFile.read(in);
        for (int i = 0; i < cf.methods.length; i++) {
            Method m = cf.methods[i];
            if (m.access_flags.is(AccessFlags.ACC_NATIVE)) {
                // System.err.println(className);
                return true;
            }
        }
        return false;
    }

    void compare(File f1, File f2) throws IOException {
        if (f1.isFile() && f2.isFile())
            compareFiles(f1, f2);
        else if (f1.isDirectory() && f2.isDirectory())
            compareDirectories(f1, f2);
        else
            error("files differ: "
                + f1 + " (" + getFileType(f1) + "), "
                + f2 + " (" + getFileType(f2) + ")");
    }

    void compareDirectories(File d1, File d2) throws IOException {
        Set<String> list = new TreeSet<String>();
        list.addAll(Arrays.asList(d1.list()));
        list.addAll(Arrays.asList(d2.list()));
        for (String c: list)
            compare(new File(d1, c), new File(d2, c));
    }

    void compareFiles(File f1, File f2) throws IOException {
        byte[] c1 = readFile(f1);
        byte[] c2 = readFile(f2);
        if (!Arrays.equals(c1, c2))
            error("files differ: " + f1 + ", " + f2);
    }

    byte[] readFile(File file) throws IOException {
        int size = (int) file.length();
        byte[] data = new byte[size];
        DataInputStream in = new DataInputStream(new FileInputStream(file));
        try {
            in.readFully(data);
        } finally {
            in.close();
        }
        return data;
    }

    String getFileType(File f) {
        return f.isDirectory() ? "directory"
                : f.isFile() ? "file"
                : f.exists() ? "other"
                : "not found";
    }

    /**
     * Set up an empty directory.
     */
    public File initDir(File dir) {
        if (dir.exists())
            deleteAll(dir);
        dir.mkdirs();
        return dir;
    }

    /**
     * Delete a file or a directory (including all its contents).
     */
    boolean deleteAll(File file) {
        if (file.isDirectory()) {
            for (File f: file.listFiles())
                deleteAll(f);
        }
        return file.delete();
    }

    File file(File dir, String... path) {
        File f = dir;
        for (String p: path)
            f = new File(f, p);
        return f;
    }

    /**
     * Report an error.
     */
    void error(String msg, String... more) {
        System.err.println("test: " + testOptions + " " + testClassName);
        System.err.println("error: " + msg);
        for (String s: more)
            System.err.println(s);
        errors++;
        System.exit(1);
    }

    File old_javah_cmd;
    File rt_jar;
    List<String> testOptions;
    String testClassName;
    int count;
    int errors;
}