mirror of
https://github.com/processing/processing4.git
synced 2026-02-03 13:49:18 +01:00
370 lines
12 KiB
Java
370 lines
12 KiB
Java
/* -*- mode: jde; c-basic-offset: 2; indent-tabs-mode: nil -*- */
|
|
|
|
/*
|
|
PdeCompiler - default compiler class that connects to jikes
|
|
Part of the Processing project - http://Proce55ing.net
|
|
|
|
Except where noted, code is written by Ben Fry and
|
|
Copyright (c) 2001-03 Massachusetts Institute of Technology
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program 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 for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program; if not, write to the Free Software Foundation,
|
|
Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*/
|
|
|
|
import java.io.*;
|
|
import java.util.*;
|
|
import java.util.zip.*;
|
|
import javax.swing.*;
|
|
|
|
public class PdeCompiler implements PdeMessageConsumer {
|
|
static final String BUGS_URL =
|
|
"http://Proce55ing.net/bugs/";
|
|
static final String SUPER_BADNESS =
|
|
"Compiler error, please submit this code to " + BUGS_URL;
|
|
|
|
String buildPath;
|
|
String className;
|
|
File includeFolder;
|
|
PdeException exception;
|
|
PdeEditor editor;
|
|
|
|
|
|
public PdeCompiler(String buildPath, String className,
|
|
File includeFolder, PdeEditor editor) {
|
|
this.buildPath = buildPath;
|
|
this.includeFolder = includeFolder;
|
|
this.className = className;
|
|
this.editor = editor;
|
|
}
|
|
|
|
|
|
public boolean compile(PrintStream leechErr) {
|
|
String userdir = System.getProperty("user.dir") + File.separator;
|
|
|
|
//System.out.println(userdir + "jikes");
|
|
//System.out.println(System.getProperty("sun.boot.class.path"));
|
|
|
|
String command[] = new String[] {
|
|
#ifdef MACOS
|
|
// linux doesn't seem to like this
|
|
// though windows probably doesn't care
|
|
userdir + "jikes",
|
|
#else
|
|
"jikes",
|
|
#endif
|
|
|
|
// necessary to make output classes compatible with 1.1
|
|
// i.e. so that exported applets can work with ms jvm on the web
|
|
"-target",
|
|
"1.1",
|
|
|
|
// used when run without a vm ("expert" mode)
|
|
"-bootclasspath",
|
|
calcBootClassPath(),
|
|
//System.getProperty("sun.boot.class.path") + additional,
|
|
|
|
// needed for macosx so that the classpath is set properly
|
|
// also for windows because qtjava will most likely be here
|
|
// and for linux, it just doesn't hurt
|
|
"-classpath",
|
|
//System.getProperty("java.class.path"),
|
|
calcClassPath(includeFolder),
|
|
|
|
"-nowarn", // we're not currently interested in warnings
|
|
"+E", // output errors in machine-parsable format
|
|
"-d", buildPath, // output the classes in the buildPath
|
|
buildPath + File.separator + className + ".java" // file to compile
|
|
};
|
|
|
|
//for (int i = 0; i < command.length; i++) {
|
|
//System.out.println("C" + i + ": " + command[i]);
|
|
//System.out.println();
|
|
//}
|
|
|
|
firstErrorFound = false; // haven't found any errors yet
|
|
secondErrorFound = false;
|
|
|
|
int result=0; // pre-initialized to quiet a bogus warning from jikes
|
|
try {
|
|
|
|
// execute the compiler, and create threads to deal with the input
|
|
// and error streams
|
|
//
|
|
Process process = Runtime.getRuntime().exec(command);
|
|
new PdeMessageSiphon(process.getInputStream(), this);
|
|
new PdeMessageSiphon(process.getErrorStream(), this);
|
|
|
|
// wait for the process to finish. if interrupted
|
|
// before waitFor returns, continue waiting
|
|
//
|
|
boolean compiling = true;
|
|
while (compiling) {
|
|
try {
|
|
result = process.waitFor();
|
|
compiling = false;
|
|
} catch (InterruptedException intExc) {
|
|
}
|
|
}
|
|
|
|
} catch (Exception e) {
|
|
String msg = e.getMessage();
|
|
if ((msg != null) && (msg.indexOf("jikes: not found") != -1)) {
|
|
//System.err.println("jikes is missing");
|
|
PdeBase.showWarning("Compiler error",
|
|
"Could not find the compiler.\n" +
|
|
"jikes is missing from your PATH,\n" +
|
|
"see readme.txt for help.", null);
|
|
return false;
|
|
}
|
|
e.printStackTrace();
|
|
result = -1;
|
|
}
|
|
|
|
// if the result isn't a known, expected value it means that something
|
|
// is fairly wrong, one possibility is that jikes has crashed.
|
|
//
|
|
if (result != 0 && result != 1 ) {
|
|
exception = new PdeException(SUPER_BADNESS);
|
|
editor.error(exception);
|
|
PdeBase.openURL(BUGS_URL);
|
|
}
|
|
|
|
return (result == 0) ? true : false;
|
|
}
|
|
|
|
|
|
boolean firstErrorFound;
|
|
boolean secondErrorFound;
|
|
|
|
/**
|
|
* Part of the PdeMessageConsumer interface, this is called
|
|
* whenever a piece (usually a line) of error message is spewed
|
|
* out from the compiler. The errors are parsed for their contents
|
|
* and line number, which is then reported back to PdeEditor.
|
|
*/
|
|
public void message(String s) {
|
|
//System.err.println("MSG: " + s);
|
|
System.err.print(s);
|
|
|
|
// ignore cautions
|
|
if (s.indexOf("Caution") != -1) return;
|
|
|
|
// jikes always uses a forward slash character as its separator,
|
|
// so replace any platform-specific separator characters before
|
|
// attemping to compare
|
|
//
|
|
String partialTempPath = buildPath.replace(File.separatorChar, '/')
|
|
+ "/" + className + ".java";
|
|
|
|
// if the partial temp path appears in the error message...
|
|
//
|
|
int partialStartIndex = s.indexOf(partialTempPath);
|
|
if (partialStartIndex != -1) {
|
|
|
|
// skip past the path and parse the int after the first colon
|
|
//
|
|
String s1 = s.substring(partialStartIndex + partialTempPath.length()
|
|
+ 1);
|
|
int colon = s1.indexOf(':');
|
|
int lineNumber = Integer.parseInt(s1.substring(0, colon));
|
|
//System.out.println("pde / line number: " + lineNumber);
|
|
|
|
//String s2 = s1.substring(colon + 2);
|
|
int err = s1.indexOf("Error:");
|
|
if (err != -1) {
|
|
|
|
// if the first error has already been found, then this must be
|
|
// (at least) the second error found
|
|
if (firstErrorFound) {
|
|
secondErrorFound = true;
|
|
return;
|
|
}
|
|
|
|
// if we're here at all, this is at least the first error
|
|
firstErrorFound = true;
|
|
|
|
//err += "error:".length();
|
|
String description = s1.substring(err + "Error:".length());
|
|
description = description.trim();
|
|
//System.out.println("description = " + description);
|
|
exception = new PdeException(description, lineNumber-1);
|
|
editor.error(exception);
|
|
|
|
} else {
|
|
System.err.println("i suck: " + s);
|
|
}
|
|
|
|
} else {
|
|
|
|
// this isn't the start of an error line, so don't attempt to parse
|
|
// a line number out of it.
|
|
|
|
// if we're not yet at the second error, these lines are probably
|
|
// associated with the first error message, which is already in the
|
|
// status bar, and are likely to be of interest to the user, so
|
|
// spit them to the console.
|
|
//
|
|
if (!secondErrorFound) {
|
|
System.err.println(s);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static String bootClassPath;
|
|
|
|
static public String calcBootClassPath() {
|
|
if (bootClassPath == null) {
|
|
if (PdeBase.platform == PdeBase.MACOSX) {
|
|
additional =
|
|
contentsToClassPath(new File("/System/Library/Java/Extensions/"));
|
|
} else {
|
|
additional = "";
|
|
}
|
|
bootClassPath = System.getProperty("sun.boot.class.path") + additional;
|
|
}
|
|
return bootClassPath;
|
|
}
|
|
|
|
|
|
/**
|
|
* Return the java.class.path, along with additional items
|
|
* that were magically gleaned from the 'include' folder,
|
|
* which may contain additional library classes.
|
|
*/
|
|
static public String calcClassPath(File include) {
|
|
return System.getProperty("java.class.path") + contentsToClassPath(include);
|
|
}
|
|
|
|
|
|
///
|
|
|
|
|
|
/**
|
|
* Return the path for a folder, with appended paths to
|
|
* any .jar or .zip files inside that folder.
|
|
* This will prepend a colon so that it can be directly
|
|
* appended to another path string.
|
|
*/
|
|
//static public String includeFolder(File folder) {
|
|
static public String contentsToClassPath(File folder) {
|
|
if (folder == null) return "";
|
|
|
|
StringBuffer abuffer = new StringBuffer();
|
|
String sep = System.getProperty("path.separator");
|
|
|
|
try {
|
|
// add the folder itself in case any unzipped files
|
|
String path = folder.getCanonicalPath();
|
|
abuffer.append(sep);
|
|
abuffer.append(path);
|
|
|
|
if (!path.endsWith(File.separator)) {
|
|
path += File.separator;
|
|
}
|
|
//System.out.println("path is " + path);
|
|
|
|
String list[] = folder.list();
|
|
for (int i = 0; i < list.length; i++) {
|
|
if (list[i].toLowerCase().endsWith(".jar") ||
|
|
list[i].toLowerCase().endsWith(".zip")) {
|
|
abuffer.append(sep);
|
|
abuffer.append(path);
|
|
abuffer.append(list[i]);
|
|
}
|
|
}
|
|
} catch (IOException e) {
|
|
e.printStackTrace(); // this would be odd
|
|
}
|
|
//System.out.println("included path is " + abuffer.toString());
|
|
magicImports(abuffer.toString());
|
|
return abuffer.toString();
|
|
}
|
|
|
|
|
|
static public String[] magicImports(String path) {
|
|
String imports[] = new String[100];
|
|
int importCount = 0;
|
|
|
|
String pieces[] =
|
|
BApplet.splitStrings(path, File.pathSeparatorChar);
|
|
|
|
for (int i = 0; i < pieces.length; i++) {
|
|
//System.out.println("checking piece " + pieces[i]);
|
|
if (pieces[i].length() == 0) continue;
|
|
|
|
if (pieces[i].toLowerCase().endsWith(".jar") ||
|
|
pieces[i].toLowerCase().endsWith(".zip")) {
|
|
try {
|
|
ZipFile file = new ZipFile(pieces[i]);
|
|
Enumeration entries = file.entries();
|
|
while (entries.hasMoreElements()) {
|
|
ZipEntry entry = (ZipEntry) entries.nextElement();
|
|
if (entry.isDirectory()) {
|
|
String name = entry.getName();
|
|
if (name.equals("META-INF/")) continue;
|
|
name = name.substring(0, name.length() - 1);
|
|
name = name.replace('/', '.');
|
|
|
|
if (importCount == imports.length) {
|
|
String temp[] = new String[importCount << 1];
|
|
System.arraycopy(imports, 0, temp, 0, importCount);
|
|
imports = temp;
|
|
}
|
|
imports[importCount++] = name;
|
|
//System.out.println("import " + name + ".*;");
|
|
}
|
|
//System.out.print(entry.isDirectory() ? "D " : "c ");
|
|
//System.out.println(entry.getName());
|
|
}
|
|
} catch (IOException e) {
|
|
System.err.println("Error in file " + pieces[i]);
|
|
e.printStackTrace();
|
|
}
|
|
} else {
|
|
File dir = new File(pieces[i]);
|
|
if (dir.exists()) {
|
|
importCount = magicImportsRecursive(dir, null,
|
|
imports, importCount);
|
|
}
|
|
}
|
|
}
|
|
String output[] = new String[importCount];
|
|
System.arraycopy(imports, 0, output, 0, importCount);
|
|
return output;
|
|
}
|
|
|
|
|
|
static public int magicImportsRecursive(File dir, String sofar,
|
|
String imports[],
|
|
int importCount) {
|
|
String files[] = dir.list();
|
|
for (int i = 0; i < files.length; i++) {
|
|
if (files[i].equals(".") || files[i].equals("..")) continue;
|
|
|
|
File sub = new File(dir, files[i]);
|
|
if (sub.isDirectory()) {
|
|
String nowfar = (sofar == null) ?
|
|
files[i] : (sofar + "." + files[i]);
|
|
//System.out.println(nowfar);
|
|
imports[importCount++] = nowfar;
|
|
|
|
importCount = magicImportsRecursive(sub, nowfar,
|
|
imports, importCount);
|
|
}
|
|
}
|
|
return importCount;
|
|
}
|
|
}
|