mirror of
https://gitlab.com/ita1024/waf.git
synced 2024-12-24 10:15:05 +01:00
432 lines
11 KiB
Python
432 lines
11 KiB
Python
#!/usr/bin/env python
|
|
# encoding: utf-8
|
|
# Thomas Nagy, 2006-2010 (ita)
|
|
|
|
"""
|
|
Java support
|
|
|
|
Javac is one of the few compilers that behaves very badly:
|
|
|
|
#. it outputs files where it wants to (-d is only for the package root)
|
|
|
|
#. it recompiles files silently behind your back
|
|
|
|
#. it outputs an undefined amount of files (inner classes)
|
|
|
|
Remember that the compilation can be performed using Jython[1] rather than regular Python. Instead of
|
|
running one of the following commands::
|
|
|
|
./waf configure
|
|
python waf configure
|
|
|
|
You would have to run::
|
|
|
|
java -jar /path/to/jython.jar waf configure
|
|
|
|
[1] http://www.jython.org/
|
|
"""
|
|
|
|
import os, re, tempfile, shutil
|
|
from waflib.Configure import conf
|
|
from waflib import TaskGen, Task, Utils, Options, Build, Errors, Node, Logs
|
|
from waflib.TaskGen import feature, before_method, after_method
|
|
|
|
from waflib.Tools import ccroot
|
|
ccroot.USELIB_VARS['javac'] = set(['CLASSPATH', 'JAVACFLAGS'])
|
|
|
|
|
|
SOURCE_RE = '**/*.java'
|
|
JAR_RE = '**/*'
|
|
|
|
class_check_source = '''
|
|
public class Test {
|
|
public static void main(String[] argv) {
|
|
Class lib;
|
|
if (argv.length < 1) {
|
|
System.err.println("Missing argument");
|
|
System.exit(77);
|
|
}
|
|
try {
|
|
lib = Class.forName(argv[0]);
|
|
} catch (ClassNotFoundException e) {
|
|
System.err.println("ClassNotFoundException");
|
|
System.exit(1);
|
|
}
|
|
lib = null;
|
|
System.exit(0);
|
|
}
|
|
}
|
|
'''
|
|
|
|
@feature('javac')
|
|
@before_method('process_source')
|
|
def apply_java(self):
|
|
"""
|
|
Create a javac task for compiling *.java files*. There can be
|
|
only one javac task by task generator.
|
|
"""
|
|
Utils.def_attrs(self, jarname='', classpath='',
|
|
sourcepath='.', srcdir='.',
|
|
jar_mf_attributes={}, jar_mf_classpath=[])
|
|
|
|
nodes_lst = []
|
|
|
|
outdir = getattr(self, 'outdir', None)
|
|
if outdir:
|
|
if not isinstance(outdir, Node.Node):
|
|
outdir = self.path.get_bld().make_node(self.outdir)
|
|
else:
|
|
outdir = self.path.get_bld()
|
|
outdir.mkdir()
|
|
self.outdir = outdir
|
|
self.env['OUTDIR'] = outdir.abspath()
|
|
|
|
self.javac_task = tsk = self.create_task('javac')
|
|
tmp = []
|
|
|
|
srcdir = getattr(self, 'srcdir', '')
|
|
if isinstance(srcdir, Node.Node):
|
|
srcdir = [srcdir]
|
|
for x in Utils.to_list(srcdir):
|
|
if isinstance(x, Node.Node):
|
|
y = x
|
|
else:
|
|
y = self.path.find_dir(x)
|
|
if not y:
|
|
self.bld.fatal('Could not find the folder %s from %s' % (x, self.path))
|
|
tmp.append(y)
|
|
tsk.srcdir = tmp
|
|
|
|
if getattr(self, 'compat', None):
|
|
tsk.env.append_value('JAVACFLAGS', ['-source', self.compat])
|
|
|
|
if hasattr(self, 'sourcepath'):
|
|
fold = [isinstance(x, Node.Node) and x or self.path.find_dir(x) for x in self.to_list(self.sourcepath)]
|
|
names = os.pathsep.join([x.srcpath() for x in fold])
|
|
else:
|
|
names = [x.srcpath() for x in tsk.srcdir]
|
|
|
|
if names:
|
|
tsk.env.append_value('JAVACFLAGS', ['-sourcepath', names])
|
|
|
|
@feature('javac')
|
|
@after_method('apply_java')
|
|
def use_javac_files(self):
|
|
"""
|
|
Process the *use* attribute referring to other java compilations
|
|
"""
|
|
lst = []
|
|
self.uselib = self.to_list(getattr(self, 'uselib', []))
|
|
names = self.to_list(getattr(self, 'use', []))
|
|
get = self.bld.get_tgen_by_name
|
|
for x in names:
|
|
try:
|
|
y = get(x)
|
|
except Exception:
|
|
self.uselib.append(x)
|
|
else:
|
|
y.post()
|
|
lst.append(y.jar_task.outputs[0].abspath())
|
|
self.javac_task.set_run_after(y.jar_task)
|
|
|
|
if lst:
|
|
self.env.append_value('CLASSPATH', lst)
|
|
|
|
@feature('javac')
|
|
@after_method('apply_java', 'propagate_uselib_vars', 'use_javac_files')
|
|
def set_classpath(self):
|
|
"""
|
|
Set the CLASSPATH value on the *javac* task previously created.
|
|
"""
|
|
self.env.append_value('CLASSPATH', getattr(self, 'classpath', []))
|
|
for x in self.tasks:
|
|
x.env.CLASSPATH = os.pathsep.join(self.env.CLASSPATH) + os.pathsep
|
|
|
|
@feature('jar')
|
|
@after_method('apply_java', 'use_javac_files')
|
|
@before_method('process_source')
|
|
def jar_files(self):
|
|
"""
|
|
Create a jar task. There can be only one jar task by task generator.
|
|
"""
|
|
destfile = getattr(self, 'destfile', 'test.jar')
|
|
jaropts = getattr(self, 'jaropts', [])
|
|
manifest = getattr(self, 'manifest', None)
|
|
|
|
basedir = getattr(self, 'basedir', None)
|
|
if basedir:
|
|
if not isinstance(self.basedir, Node.Node):
|
|
basedir = self.path.get_bld().make_node(basedir)
|
|
else:
|
|
basedir = self.path.get_bld()
|
|
if not basedir:
|
|
self.bld.fatal('Could not find the basedir %r for %r' % (self.basedir, self))
|
|
|
|
self.jar_task = tsk = self.create_task('jar_create')
|
|
if manifest:
|
|
jarcreate = getattr(self, 'jarcreate', 'cfm')
|
|
node = self.path.find_node(manifest)
|
|
tsk.dep_nodes.append(node)
|
|
jaropts.insert(0, node.abspath())
|
|
else:
|
|
jarcreate = getattr(self, 'jarcreate', 'cf')
|
|
if not isinstance(destfile, Node.Node):
|
|
destfile = self.path.find_or_declare(destfile)
|
|
if not destfile:
|
|
self.bld.fatal('invalid destfile %r for %r' % (destfile, self))
|
|
tsk.set_outputs(destfile)
|
|
tsk.basedir = basedir
|
|
|
|
jaropts.append('-C')
|
|
jaropts.append(basedir.bldpath())
|
|
jaropts.append('.')
|
|
|
|
tsk.env['JAROPTS'] = jaropts
|
|
tsk.env['JARCREATE'] = jarcreate
|
|
|
|
if getattr(self, 'javac_task', None):
|
|
tsk.set_run_after(self.javac_task)
|
|
|
|
@feature('jar')
|
|
@after_method('jar_files')
|
|
def use_jar_files(self):
|
|
"""
|
|
Process the *use* attribute to set the build order on the
|
|
tasks created by another task generator.
|
|
"""
|
|
lst = []
|
|
self.uselib = self.to_list(getattr(self, 'uselib', []))
|
|
names = self.to_list(getattr(self, 'use', []))
|
|
get = self.bld.get_tgen_by_name
|
|
for x in names:
|
|
try:
|
|
y = get(x)
|
|
except Exception:
|
|
self.uselib.append(x)
|
|
else:
|
|
y.post()
|
|
self.jar_task.run_after.update(y.tasks)
|
|
|
|
class jar_create(Task.Task):
|
|
"""
|
|
Create a jar file
|
|
"""
|
|
color = 'GREEN'
|
|
run_str = '${JAR} ${JARCREATE} ${TGT} ${JAROPTS}'
|
|
|
|
def runnable_status(self):
|
|
"""
|
|
Wait for dependent tasks to be executed, then read the
|
|
files to update the list of inputs.
|
|
"""
|
|
for t in self.run_after:
|
|
if not t.hasrun:
|
|
return Task.ASK_LATER
|
|
if not self.inputs:
|
|
global JAR_RE
|
|
try:
|
|
self.inputs = [x for x in self.basedir.ant_glob(JAR_RE, remove=False) if id(x) != id(self.outputs[0])]
|
|
except Exception:
|
|
raise Errors.WafError('Could not find the basedir %r for %r' % (self.basedir, self))
|
|
return super(jar_create, self).runnable_status()
|
|
|
|
class javac(Task.Task):
|
|
"""
|
|
Compile java files
|
|
"""
|
|
color = 'BLUE'
|
|
|
|
nocache = True
|
|
"""
|
|
The .class files cannot be put into a cache at the moment
|
|
"""
|
|
|
|
vars = ['CLASSPATH', 'JAVACFLAGS', 'JAVAC', 'OUTDIR']
|
|
"""
|
|
The javac task will be executed again if the variables CLASSPATH, JAVACFLAGS, JAVAC or OUTDIR change.
|
|
"""
|
|
|
|
def runnable_status(self):
|
|
"""
|
|
Wait for dependent tasks to be complete, then read the file system to find the input nodes.
|
|
"""
|
|
for t in self.run_after:
|
|
if not t.hasrun:
|
|
return Task.ASK_LATER
|
|
|
|
if not self.inputs:
|
|
global SOURCE_RE
|
|
self.inputs = []
|
|
for x in self.srcdir:
|
|
self.inputs.extend(x.ant_glob(SOURCE_RE, remove=False))
|
|
return super(javac, self).runnable_status()
|
|
|
|
def run(self):
|
|
"""
|
|
Execute the javac compiler
|
|
"""
|
|
env = self.env
|
|
gen = self.generator
|
|
bld = gen.bld
|
|
wd = bld.bldnode.abspath()
|
|
def to_list(xx):
|
|
if isinstance(xx, str): return [xx]
|
|
return xx
|
|
cmd = []
|
|
cmd.extend(to_list(env['JAVAC']))
|
|
cmd.extend(['-classpath'])
|
|
cmd.extend(to_list(env['CLASSPATH']))
|
|
cmd.extend(['-d'])
|
|
cmd.extend(to_list(env['OUTDIR']))
|
|
cmd.extend(to_list(env['JAVACFLAGS']))
|
|
|
|
files = [a.path_from(bld.bldnode) for a in self.inputs]
|
|
|
|
# workaround for command line length limit:
|
|
# http://support.microsoft.com/kb/830473
|
|
tmp = None
|
|
try:
|
|
if len(str(files)) + len(str(cmd)) > 8192:
|
|
(fd, tmp) = tempfile.mkstemp(dir=bld.bldnode.abspath())
|
|
try:
|
|
os.write(fd, '\n'.join(files).encode())
|
|
finally:
|
|
if tmp:
|
|
os.close(fd)
|
|
if Logs.verbose:
|
|
Logs.debug('runner: %r' % (cmd + files))
|
|
cmd.append('@' + tmp)
|
|
else:
|
|
cmd += files
|
|
|
|
ret = self.exec_command(cmd, cwd=wd, env=env.env or None)
|
|
finally:
|
|
if tmp:
|
|
os.unlink(tmp)
|
|
return ret
|
|
|
|
def post_run(self):
|
|
"""
|
|
"""
|
|
for n in self.generator.outdir.ant_glob('**/*.class'):
|
|
n.sig = Utils.h_file(n.abspath()) # careful with this
|
|
self.generator.bld.task_sigs[self.uid()] = self.cache_sig
|
|
|
|
def configure(self):
|
|
"""
|
|
Detect the javac, java and jar programs
|
|
"""
|
|
# If JAVA_PATH is set, we prepend it to the path list
|
|
java_path = self.environ['PATH'].split(os.pathsep)
|
|
v = self.env
|
|
|
|
if 'JAVA_HOME' in self.environ:
|
|
java_path = [os.path.join(self.environ['JAVA_HOME'], 'bin')] + java_path
|
|
self.env['JAVA_HOME'] = [self.environ['JAVA_HOME']]
|
|
|
|
for x in 'javac java jar'.split():
|
|
self.find_program(x, var=x.upper(), path_list=java_path)
|
|
self.env[x.upper()] = self.cmd_to_list(self.env[x.upper()])
|
|
|
|
if 'CLASSPATH' in self.environ:
|
|
v['CLASSPATH'] = self.environ['CLASSPATH']
|
|
|
|
if not v['JAR']: self.fatal('jar is required for making java packages')
|
|
if not v['JAVAC']: self.fatal('javac is required for compiling java classes')
|
|
v['JARCREATE'] = 'cf' # can use cvf
|
|
v['JAVACFLAGS'] = []
|
|
|
|
@conf
|
|
def check_java_class(self, classname, with_classpath=None):
|
|
"""
|
|
Check if the specified java class exists
|
|
|
|
:param classname: class to check, like java.util.HashMap
|
|
:type classname: string
|
|
:param with_classpath: additional classpath to give
|
|
:type with_classpath: string
|
|
"""
|
|
|
|
javatestdir = '.waf-javatest'
|
|
|
|
classpath = javatestdir
|
|
if self.env['CLASSPATH']:
|
|
classpath += os.pathsep + self.env['CLASSPATH']
|
|
if isinstance(with_classpath, str):
|
|
classpath += os.pathsep + with_classpath
|
|
|
|
shutil.rmtree(javatestdir, True)
|
|
os.mkdir(javatestdir)
|
|
|
|
java_file = open(os.path.join(javatestdir, 'Test.java'), 'w')
|
|
java_file.write(class_check_source)
|
|
java_file.close()
|
|
|
|
# Compile the source
|
|
self.exec_command(self.env['JAVAC'] + [os.path.join(javatestdir, 'Test.java')], shell=False)
|
|
|
|
# Try to run the app
|
|
cmd = self.env['JAVA'] + ['-cp', classpath, 'Test', classname]
|
|
self.to_log("%s\n" % str(cmd))
|
|
found = self.exec_command(cmd, shell=False)
|
|
|
|
self.msg('Checking for java class %s' % classname, not found)
|
|
|
|
shutil.rmtree(javatestdir, True)
|
|
|
|
return found
|
|
|
|
@conf
|
|
def check_jni_headers(conf):
|
|
"""
|
|
Check for jni headers and libraries. On success the conf.env variables xxx_JAVA are added for use in C/C++ targets::
|
|
|
|
def options(opt):
|
|
opt.load('compiler_c')
|
|
|
|
def configure(conf):
|
|
conf.load('compiler_c java')
|
|
conf.check_jni_headers()
|
|
|
|
def build(bld):
|
|
bld.shlib(source='a.c', target='app', use='JAVA')
|
|
"""
|
|
|
|
if not conf.env.CC_NAME and not conf.env.CXX_NAME:
|
|
conf.fatal('load a compiler first (gcc, g++, ..)')
|
|
|
|
if not conf.env.JAVA_HOME:
|
|
conf.fatal('set JAVA_HOME in the system environment')
|
|
|
|
# jni requires the jvm
|
|
javaHome = conf.env['JAVA_HOME'][0]
|
|
|
|
dir = conf.root.find_dir(conf.env.JAVA_HOME[0] + '/include')
|
|
if dir is None:
|
|
conf.fatal('JAVA_HOME does not seem to be set properly')
|
|
|
|
f = dir.ant_glob('**/(jni|jni_md).h')
|
|
incDirs = [x.parent.abspath() for x in f]
|
|
|
|
dir = conf.root.find_dir(conf.env.JAVA_HOME[0])
|
|
f = dir.ant_glob('**/*jvm.(so|dll|dylib)')
|
|
libDirs = [x.parent.abspath() for x in f] or [javaHome]
|
|
|
|
# On windows, we need both the .dll and .lib to link. On my JDK, they are
|
|
# in different directories...
|
|
f = dir.ant_glob('**/*jvm.(lib)')
|
|
if f:
|
|
libDirs = [[x, y.parent.abspath()] for x in libDirs for y in f]
|
|
|
|
for d in libDirs:
|
|
try:
|
|
conf.check(header_name='jni.h', define_name='HAVE_JNI_H', lib='jvm',
|
|
libpath=d, includes=incDirs, uselib_store='JAVA', uselib='JAVA')
|
|
except Exception:
|
|
pass
|
|
else:
|
|
break
|
|
else:
|
|
conf.fatal('could not find lib jvm in %r (see config.log)' % libDirs)
|
|
|