2008-02-19 11:06:52 -07:00
|
|
|
Quick Start
|
|
|
|
-----------
|
|
|
|
|
2008-02-20 09:55:43 -07:00
|
|
|
on Linux:
|
2008-03-10 07:27:42 -06:00
|
|
|
$ export JAVA_HOME=/usr/local/java # or wherever you have the JDK installed
|
2008-02-19 11:06:52 -07:00
|
|
|
$ make
|
2009-03-15 13:49:13 -06:00
|
|
|
$ build/linux-i386/avian -cp build/test Hello
|
2008-02-19 11:06:52 -07:00
|
|
|
|
2008-02-20 09:55:43 -07:00
|
|
|
on Mac OS X:
|
2008-02-20 09:41:55 -07:00
|
|
|
$ export JAVA_HOME=/Library/Java/Home
|
|
|
|
$ make
|
2009-03-15 13:49:13 -06:00
|
|
|
$ build/darwin-i386/avian -cp build/test Hello
|
2008-10-10 08:06:31 -06:00
|
|
|
|
2008-10-11 14:46:20 -06:00
|
|
|
on Windows (MSYS):
|
2009-12-02 18:37:22 -07:00
|
|
|
$ git clone git://oss.readytalk.com/win32.git ../win32
|
2008-10-10 11:47:31 -06:00
|
|
|
$ export JAVA_HOME="C:/Program Files/Java/jdk1.6.0_07"
|
2008-10-10 08:06:31 -06:00
|
|
|
$ make
|
2009-03-15 13:49:13 -06:00
|
|
|
$ build/windows-i386/avian -cp build/test Hello
|
2008-10-10 11:47:31 -06:00
|
|
|
|
2008-10-11 14:46:20 -06:00
|
|
|
on Windows (Cygwin):
|
2009-12-02 18:37:22 -07:00
|
|
|
$ git clone git://oss.readytalk.com/win32.git ../win32
|
2008-10-11 14:46:20 -06:00
|
|
|
$ export JAVA_HOME="/cygdrive/c/Program Files/Java/jdk1.6.0_07"
|
|
|
|
$ make
|
2009-03-15 13:49:13 -06:00
|
|
|
$ build/windows-i386/avian -cp build/test Hello
|
2008-10-11 14:46:20 -06:00
|
|
|
|
2008-10-10 11:47:31 -06:00
|
|
|
Adjust JAVA_HOME according to your system, but be sure to use forward
|
|
|
|
slashes in the path.
|
|
|
|
|
2008-02-20 10:57:52 -07:00
|
|
|
|
2008-03-06 13:29:25 -07:00
|
|
|
Introduction
|
|
|
|
------------
|
|
|
|
|
|
|
|
Avian is a lightweight virtual machine and class library designed to
|
|
|
|
provide a useful subset of Java's features, suitable for building
|
|
|
|
self-contained applications. More information is available at the
|
|
|
|
project web site:
|
|
|
|
|
|
|
|
http://oss.readytalk.com/avian
|
|
|
|
|
|
|
|
If you have any trouble building, running, or embedding Avian, please
|
|
|
|
post a message to our discussion group:
|
|
|
|
|
|
|
|
http://groups.google.com/group/avian
|
|
|
|
|
|
|
|
That's also the place for any other questions, comments, or
|
|
|
|
suggestions you might have.
|
|
|
|
|
|
|
|
|
2008-02-19 11:06:52 -07:00
|
|
|
Supported Platforms
|
|
|
|
-------------------
|
|
|
|
|
|
|
|
Avian can currently target the following platforms:
|
|
|
|
|
|
|
|
Linux (i386 and x86_64)
|
2009-10-27 09:26:19 -06:00
|
|
|
Windows (i386 and x86_64)
|
2009-10-14 10:01:37 -06:00
|
|
|
Mac OS X (i386, x86_64 and 32-bit PowerPC)
|
2008-02-19 11:06:52 -07:00
|
|
|
|
|
|
|
|
|
|
|
Building
|
|
|
|
--------
|
|
|
|
|
2008-03-06 13:29:25 -07:00
|
|
|
Build requirements include:
|
|
|
|
|
|
|
|
* GNU make 3.80 or later
|
2009-10-27 09:22:02 -06:00
|
|
|
* GCC 3.4 or later (4.5 or later for Windows/x86_64)
|
2008-03-06 13:29:25 -07:00
|
|
|
* JDK 1.5 or later
|
2009-03-15 13:49:13 -06:00
|
|
|
* MinGW 3.4 or later (only if compiling for Windows)
|
2008-03-06 13:29:25 -07:00
|
|
|
* zlib 1.2.3 or later
|
|
|
|
|
|
|
|
Earlier versions of some of these packages may also work but have not
|
|
|
|
been tested.
|
|
|
|
|
2008-02-19 17:09:37 -07:00
|
|
|
The build is directed by a single makefile and may be influenced via
|
2009-03-15 13:49:13 -06:00
|
|
|
certain flags described below, all of which are optional.
|
2008-02-19 17:09:37 -07:00
|
|
|
|
2009-03-15 13:49:13 -06:00
|
|
|
$ make platform={linux,windows,darwin} arch={i386,x86_64,powerpc} \
|
|
|
|
process={compile,interpret} mode={debug,debug-fast,fast,small} \
|
2009-05-31 18:31:15 -06:00
|
|
|
bootimage={true,false} heapdump={true,false} tails={true,false} \
|
|
|
|
continuations={true,false}
|
2008-02-19 17:09:37 -07:00
|
|
|
|
|
|
|
* platform - the target platform
|
2009-03-15 13:49:13 -06:00
|
|
|
default: output of $(uname -s | tr [:upper:] [:lower:]),
|
|
|
|
normalized in some cases (e.g. CYGWIN_NT-5.1 -> windows)
|
2008-02-19 17:09:37 -07:00
|
|
|
|
|
|
|
* arch - the target architecture
|
2009-03-15 13:49:13 -06:00
|
|
|
default: output of $(uname -m), normalized in some cases
|
|
|
|
(e.g. i686 -> i386)
|
2008-02-19 11:06:52 -07:00
|
|
|
|
2009-05-31 18:31:15 -06:00
|
|
|
* mode - which set of compilation flags to use to determine
|
2008-02-19 17:09:37 -07:00
|
|
|
optimization level, debug symbols, and whether to enable
|
2008-03-06 13:46:44 -07:00
|
|
|
assertions
|
2008-02-19 17:09:37 -07:00
|
|
|
default: fast
|
2008-02-19 11:06:52 -07:00
|
|
|
|
2008-02-19 17:09:37 -07:00
|
|
|
* process - choice between pure interpreter or JIT compiler
|
|
|
|
default: compile
|
2008-02-19 11:06:52 -07:00
|
|
|
|
2009-03-15 13:49:13 -06:00
|
|
|
* bootimage - if true, create a boot image containing the pre-parsed
|
|
|
|
class library and ahead-of-time compiled methods. This option is
|
2009-12-03 10:47:08 -07:00
|
|
|
only valid for process=compile builds. Note that you may need to
|
|
|
|
specify both build-arch=x86_64 and arch=x86_64 on 64-bit systems
|
|
|
|
where "uname -m" prints "i386".
|
2009-03-15 13:49:13 -06:00
|
|
|
default: false
|
|
|
|
|
2009-09-15 07:59:24 -06:00
|
|
|
* heapdump - if true, implement avian.Machine.dumpHeap(String),
|
2009-03-15 13:49:13 -06:00
|
|
|
which, when called, will generate a snapshot of the heap in a
|
|
|
|
simple, ad-hoc format for memory profiling purposes. See
|
|
|
|
heapdump.cpp for details.
|
|
|
|
default: false
|
|
|
|
|
2009-05-31 18:31:15 -06:00
|
|
|
* tails - if true, optimize each tail call by replacing the caller's
|
|
|
|
stack frame with the callee's. This convention ensures proper
|
|
|
|
tail recursion, suitable for languages such as Scheme. This
|
|
|
|
option is only valid for process=compile builds.
|
|
|
|
default: false
|
|
|
|
|
|
|
|
* continuations - if true, support continuations via the
|
|
|
|
avian.Continuations methods callWithCurrentContinuation and
|
|
|
|
dynamicWind. See Continuations.java for details. This option is
|
|
|
|
only valid for process=compile builds.
|
|
|
|
default: false
|
|
|
|
|
2009-03-15 13:49:13 -06:00
|
|
|
These flags determine the name of the directory used for the build.
|
|
|
|
The name always starts with ${platform}-${arch}, and each non-default
|
|
|
|
build option is appended to the name. For example, a debug build with
|
|
|
|
bootimage enabled on Linux/i386 would be built in
|
|
|
|
build/linux-i386-debug-bootimage. This allows you to build with
|
|
|
|
several different sets of options independently and even
|
|
|
|
simultaneously without doing a clean build each time.
|
|
|
|
|
2008-10-11 14:46:20 -06:00
|
|
|
If you are compiling for Windows, you may either cross-compile using
|
|
|
|
MinGW or build natively on Windows under MSYS or Cygwin.
|
|
|
|
|
|
|
|
Installing MSYS:
|
|
|
|
|
|
|
|
1. Download and install the current MinGW and MSYS packages from
|
|
|
|
mingw.org, selecting the C and C++ compilers when prompted. Use the
|
|
|
|
post-install script to create the filesystem link to the compiler.
|
|
|
|
|
|
|
|
2. Download GNU Make 3.81 from the MSYS download page
|
|
|
|
(make-3.81-MSYS-1.0.11-2.tar.bz2) and extract the tar file into
|
|
|
|
e.g. c:/msys/1.0.
|
|
|
|
|
|
|
|
Installing Cygwin:
|
|
|
|
|
|
|
|
1. Download and run setup.exe from cygwin.com, installing the base
|
|
|
|
system and these packages: make, gcc-mingw-g++, and (optionally)
|
|
|
|
git.
|
|
|
|
|
|
|
|
You may also find our win32 repository useful: (run this from the
|
|
|
|
directory containing the avian directory)
|
|
|
|
|
|
|
|
$ git clone git://oss.readytalk.com/win32.git
|
|
|
|
|
|
|
|
This gives you the Windows JNI headers, zlib headers and library, and
|
2009-03-15 13:49:13 -06:00
|
|
|
a few other useful libraries like OpenSSL, libjpeg, and libpng.
|
2009-12-02 18:37:22 -07:00
|
|
|
There's also a win64 repository for 64-bit builds:
|
|
|
|
|
|
|
|
$ git clone git://oss.readytalk.com/win64.git
|
|
|
|
|
|
|
|
|
2010-01-15 08:25:06 -07:00
|
|
|
Building with the Microsoft Visual C++ Compiler
|
2009-12-02 18:37:22 -07:00
|
|
|
-----------------------------------------------
|
|
|
|
|
|
|
|
You can also build using the MSVC compiler, which makes debugging with
|
|
|
|
tools like WinDbg and Visual Studio much easier. Note that you will
|
|
|
|
still need to have GCC installed - MSVC is only used to compile the
|
|
|
|
C++ portions of the VM, while the assembly code and helper tools are
|
|
|
|
built using GCC.
|
|
|
|
|
|
|
|
The MSVC build has been tested with Visual Studio Express Edition
|
|
|
|
versions 8 and 9. Other versions may also work.
|
|
|
|
|
|
|
|
To build with MSVC, install Cygwin as described above and set the
|
|
|
|
following environment variables:
|
|
|
|
|
|
|
|
$ export PATH="/usr/local/bin:/usr/bin:/bin:/usr/X11R6/bin:/cygdrive/c/Program Files/Microsoft Visual Studio 9.0/Common7/IDE:/cygdrive/c/Program Files/Microsoft Visual Studio 9.0/VC/BIN:/cygdrive/c/Program Files/Microsoft Visual Studio 9.0/Common7/Tools:/cygdrive/c/WINDOWS/Microsoft.NET/Framework/v3.5:/cygdrive/c/WINDOWS/Microsoft.NET/Framework/v2.0.50727:/cygdrive/c/Program Files/Microsoft Visual Studio 9.0/VC/VCPackages:/cygdrive/c/Program Files/Microsoft SDKs/Windows/v6.0A/bin:/cygdrive/c/WINDOWS/system32:/cygdrive/c/WINDOWS:/cygdrive/c/WINDOWS/System32/Wbem"
|
|
|
|
|
|
|
|
$ export LIBPATH="C:\WINDOWS\Microsoft.NET\Framework\v3.5;C:\WINDOWS\Microsoft.NET\Framework\v2.0.50727;C:\Program Files\Microsoft Visual Studio 9.0\VC\LIB;"
|
|
|
|
|
|
|
|
$ export VCINSTALLDIR="C:\Program Files\Microsoft Visual Studio 9.0\VC"
|
|
|
|
|
|
|
|
$ export LIB="C:\Program Files\Microsoft Visual Studio 9.0\VC\LIB;C:\Program Files\Microsoft SDKs\Windows\v6.0A\lib;"
|
|
|
|
|
|
|
|
$ export INCLUDE="C:\Program Files\Microsoft Visual Studio 9.0\VC\INCLUDE;C:\Program Files\Microsoft SDKs\Windows\v6.0A\include;"
|
|
|
|
|
|
|
|
Adjust these definitions as necessary according to your MSVC
|
|
|
|
installation.
|
|
|
|
|
|
|
|
Finally, build with the msvc flag set to the MSVC tool directory:
|
|
|
|
|
|
|
|
$ make msvc="/cygdrive/c/Program Files/Microsoft Visual Studio 9.0/VC"
|
2008-10-11 14:46:20 -06:00
|
|
|
|
2008-02-19 11:06:52 -07:00
|
|
|
|
2009-12-05 20:05:54 -07:00
|
|
|
Building with GNU Classpath
|
|
|
|
---------------------------
|
|
|
|
|
|
|
|
** Please note that this feature is still under development and is
|
|
|
|
neither complete nor well-tested. **
|
|
|
|
|
|
|
|
By default, Avian uses its own lightweight class library. However,
|
|
|
|
that library only contains a relatively small subset of the classes
|
|
|
|
and methods included in the JRE. If your application requires
|
|
|
|
features beyond that subset, you may want to tell Avian to use GNU
|
2009-12-06 13:35:00 -07:00
|
|
|
Classpath instead. To do so, specify the directory where Classpath is
|
|
|
|
installed, e.g.:
|
2009-12-05 20:05:54 -07:00
|
|
|
|
|
|
|
$ make clean
|
|
|
|
$ make gnu=/usr/local/classpath-0.98
|
|
|
|
|
|
|
|
This build will use the classes and native code from Classpath, except
|
|
|
|
that certain core classes are replaced with implementations from the
|
|
|
|
Avian class library for compatibility with the VM.
|
|
|
|
|
|
|
|
|
2008-02-19 11:06:52 -07:00
|
|
|
Installing
|
|
|
|
----------
|
|
|
|
|
2009-03-15 13:49:13 -06:00
|
|
|
Installing Avian is as simple as copying the executable to the desired
|
|
|
|
directory:
|
|
|
|
|
|
|
|
$ cp build/${platform}-${arch}/avian ~/bin/
|
2008-02-19 17:09:37 -07:00
|
|
|
|
|
|
|
|
|
|
|
Embedding
|
|
|
|
---------
|
|
|
|
|
|
|
|
The following series of commands illustrates how to produce a
|
|
|
|
stand-alone executable out of a Java application using Avian.
|
|
|
|
|
2008-10-11 14:46:20 -06:00
|
|
|
Note: if you are building on Cygwin, add -mno-cygwin to each of the
|
|
|
|
compile and link commands below.
|
|
|
|
|
2008-02-19 18:53:47 -07:00
|
|
|
Step 1: Build Avian, create a new directory, and populate it with the
|
|
|
|
VM object files and bootstrap classpath jar.
|
|
|
|
|
|
|
|
$ make
|
|
|
|
$ mkdir hello
|
|
|
|
$ cd hello
|
2009-03-15 13:49:13 -06:00
|
|
|
$ ar x ../build/${platform}-${arch}/libavian.a
|
2008-02-19 19:17:15 -07:00
|
|
|
$ cp ../build/classpath.jar boot.jar
|
2008-02-19 18:53:47 -07:00
|
|
|
|
|
|
|
Step 2: Build the Java code and add it to the jar.
|
2008-02-19 17:09:37 -07:00
|
|
|
|
|
|
|
$ cat >Hello.java <<EOF
|
|
|
|
public class Hello {
|
|
|
|
public static void main(String[] args) {
|
|
|
|
System.out.println("hello, world!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
EOF
|
2008-02-20 09:41:55 -07:00
|
|
|
$ javac -bootclasspath boot.jar Hello.java
|
2008-02-21 08:33:02 -07:00
|
|
|
$ jar u0f boot.jar Hello.class
|
2008-02-19 17:09:37 -07:00
|
|
|
|
2008-02-19 18:53:47 -07:00
|
|
|
Step 3: Make an object file out of the jar.
|
2008-02-19 17:09:37 -07:00
|
|
|
|
2009-10-13 19:00:16 -06:00
|
|
|
$ ../build/${platform}-${arch}/binaryToObject boot.jar boot-jar.o \
|
|
|
|
_binary_boot_jar_start _binary_boot_jar_end ${platform} ${arch}
|
2008-02-19 17:09:37 -07:00
|
|
|
|
2008-02-19 18:53:47 -07:00
|
|
|
Step 4: Write a driver which starts the VM and runs the desired main
|
|
|
|
method. Note the bootJar function, which will be called by the VM to
|
2008-02-19 17:09:37 -07:00
|
|
|
get a handle to the embedded jar. We tell the VM about this jar by
|
2008-02-19 18:53:47 -07:00
|
|
|
setting the classpath to "[bootJar]".
|
2008-02-19 17:09:37 -07:00
|
|
|
|
|
|
|
$ cat >main.cpp <<EOF
|
|
|
|
#include "stdint.h"
|
|
|
|
#include "jni.h"
|
|
|
|
|
|
|
|
#ifdef __MINGW32__
|
|
|
|
# define EXPORT __declspec(dllexport)
|
2008-02-19 18:53:47 -07:00
|
|
|
# define SYMBOL(x) binary_boot_jar_##x
|
2008-02-19 17:09:37 -07:00
|
|
|
#else
|
|
|
|
# define EXPORT __attribute__ ((visibility("default")))
|
2008-02-19 18:53:47 -07:00
|
|
|
# define SYMBOL(x) _binary_boot_jar_##x
|
2008-02-19 17:09:37 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
|
|
|
|
extern const uint8_t SYMBOL(start)[];
|
2008-03-30 21:43:43 -06:00
|
|
|
extern const uint8_t SYMBOL(end)[];
|
2008-02-19 17:09:37 -07:00
|
|
|
|
|
|
|
EXPORT const uint8_t*
|
2008-02-19 18:53:47 -07:00
|
|
|
bootJar(unsigned* size)
|
2008-02-19 17:09:37 -07:00
|
|
|
{
|
2008-03-30 21:43:43 -06:00
|
|
|
*size = SYMBOL(end) - SYMBOL(start);
|
2008-02-19 17:09:37 -07:00
|
|
|
return SYMBOL(start);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // extern "C"
|
|
|
|
|
|
|
|
int
|
|
|
|
main(int ac, const char** av)
|
|
|
|
{
|
2008-07-14 11:02:43 -06:00
|
|
|
JavaVMInitArgs vmArgs;
|
|
|
|
vmArgs.version = JNI_VERSION_1_2;
|
|
|
|
vmArgs.nOptions = 1;
|
|
|
|
vmArgs.ignoreUnrecognized = JNI_TRUE;
|
|
|
|
|
|
|
|
JavaVMOption options[vmArgs.nOptions];
|
|
|
|
vmArgs.options = options;
|
2008-02-19 17:09:37 -07:00
|
|
|
|
2008-07-14 11:02:43 -06:00
|
|
|
options[0].optionString = const_cast<char*>("-Djava.class.path=[bootJar]");
|
2008-02-19 17:09:37 -07:00
|
|
|
|
|
|
|
JavaVM* vm;
|
|
|
|
void* env;
|
|
|
|
JNI_CreateJavaVM(&vm, &env, &vmArgs);
|
|
|
|
JNIEnv* e = static_cast<JNIEnv*>(env);
|
|
|
|
|
2008-02-19 18:53:47 -07:00
|
|
|
jclass c = e->FindClass("Hello");
|
2009-12-16 19:25:03 -07:00
|
|
|
if (not e->ExceptionCheck()) {
|
2008-02-19 17:09:37 -07:00
|
|
|
jmethodID m = e->GetStaticMethodID(c, "main", "([Ljava/lang/String;)V");
|
2009-12-16 19:25:03 -07:00
|
|
|
if (not e->ExceptionCheck()) {
|
2008-02-19 17:09:37 -07:00
|
|
|
jclass stringClass = e->FindClass("java/lang/String");
|
2009-12-16 19:25:03 -07:00
|
|
|
if (not e->ExceptionCheck()) {
|
2008-02-19 17:09:37 -07:00
|
|
|
jobjectArray a = e->NewObjectArray(ac-1, stringClass, 0);
|
2009-12-16 19:25:03 -07:00
|
|
|
if (not e->ExceptionCheck()) {
|
2008-02-19 17:09:37 -07:00
|
|
|
for (int i = 1; i < ac; ++i) {
|
|
|
|
e->SetObjectArrayElement(a, i-1, e->NewStringUTF(av[i]));
|
|
|
|
}
|
|
|
|
|
|
|
|
e->CallStaticVoidMethod(c, m, a);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int exitCode = 0;
|
2009-12-16 19:25:03 -07:00
|
|
|
if (e->ExceptionCheck()) {
|
2008-02-19 17:09:37 -07:00
|
|
|
exitCode = -1;
|
|
|
|
e->ExceptionDescribe();
|
|
|
|
}
|
|
|
|
|
|
|
|
vm->DestroyJavaVM();
|
|
|
|
|
|
|
|
return exitCode;
|
|
|
|
}
|
|
|
|
EOF
|
2008-10-10 18:08:01 -06:00
|
|
|
|
|
|
|
on Linux:
|
|
|
|
$ g++ -I$JAVA_HOME/include -I$JAVA_HOME/include/linux \
|
|
|
|
-D_JNI_IMPLEMENTATION_ -c main.cpp -o main.o
|
|
|
|
|
|
|
|
on Mac OS X:
|
2008-10-10 11:47:31 -06:00
|
|
|
$ g++ -I$JAVA_HOME/include -D_JNI_IMPLEMENTATION_ -c main.cpp -o main.o
|
2008-02-19 17:09:37 -07:00
|
|
|
|
2008-10-10 18:08:01 -06:00
|
|
|
on Windows:
|
|
|
|
$ g++ -I$JAVA_HOME/include -I$JAVA_HOME/include/win32 \
|
|
|
|
-D_JNI_IMPLEMENTATION_ -c main.cpp -o main.o
|
2008-10-10 17:53:51 -06:00
|
|
|
|
2008-02-19 18:53:47 -07:00
|
|
|
Step 5: Link the objects produced above to produce the final
|
|
|
|
executable, and optionally strip its symbols.
|
2008-02-19 17:09:37 -07:00
|
|
|
|
2008-02-20 09:55:43 -07:00
|
|
|
on Linux:
|
2008-02-19 19:01:26 -07:00
|
|
|
$ g++ -rdynamic *.o -ldl -lpthread -lz -o hello
|
2008-02-19 18:53:47 -07:00
|
|
|
$ strip --strip-all hello
|
2008-02-20 09:55:43 -07:00
|
|
|
|
|
|
|
on Mac OS X:
|
2008-02-20 09:41:55 -07:00
|
|
|
$ g++ -rdynamic *.o -ldl -lpthread -lz -o hello -framework CoreFoundation
|
|
|
|
$ strip -S -x hello
|
|
|
|
|
2008-10-10 08:06:31 -06:00
|
|
|
on Windows:
|
|
|
|
$ dlltool -z hello.def *.o
|
2008-10-10 17:37:36 -06:00
|
|
|
$ dlltool -d hello.def -e hello.exp
|
2008-10-10 11:47:31 -06:00
|
|
|
$ g++ hello.exp *.o -L../../win32/lib -lmingwthrd -lm -lz -lws2_32 \
|
2008-10-10 17:53:51 -06:00
|
|
|
-mwindows -mconsole -o hello.exe
|
2008-10-10 08:06:31 -06:00
|
|
|
$ strip --strip-all hello.exe
|
2009-03-15 13:49:13 -06:00
|
|
|
|
|
|
|
|
|
|
|
Embedding with ProGuard and a Boot Image
|
|
|
|
----------------------------------------
|
|
|
|
|
|
|
|
The following illustrates how to embed an application as above, except
|
|
|
|
this time we preprocess the code using ProGuard and build a boot image
|
|
|
|
from it for quicker startup. The pros and cons of using ProGuard are
|
|
|
|
as follow:
|
|
|
|
|
|
|
|
* Pros: ProGuard will eliminate unused code, optimize the rest, and
|
|
|
|
obfuscate it as well for maximum space savings
|
|
|
|
|
|
|
|
* Cons: increased build time, especially for large applications, and
|
|
|
|
extra effort needed to configure it for applications which rely
|
|
|
|
heavily on reflection and/or calls to Java from native code
|
|
|
|
|
|
|
|
For boot image builds:
|
|
|
|
|
|
|
|
* Pros: the boot image build pre-parses all the classes and compiles
|
|
|
|
all the methods, obviating the need for JIT compilation at runtime.
|
|
|
|
This also makes garbage collection faster, since the pre-parsed
|
|
|
|
classes are never visited.
|
|
|
|
|
|
|
|
* Cons: the pre-parsed classes and AOT-compiled methods take up more
|
|
|
|
space in the executable than the equivalent class files. In
|
2009-03-15 17:29:45 -06:00
|
|
|
practice, this can make the executable 30-50% larger. Also, AOT
|
2009-03-15 13:49:13 -06:00
|
|
|
compilation does not yet yield significantly faster or smaller code
|
|
|
|
than JIT compilation.
|
|
|
|
|
|
|
|
Note you can use ProGuard without using a boot image and vice-versa,
|
|
|
|
as desired.
|
|
|
|
|
|
|
|
The following instructions assume we are building for Linux/i386.
|
|
|
|
Please refer to the previous example for guidance on other platforms.
|
|
|
|
|
|
|
|
Step 1: Build Avian, create a new directory, and populate it with the
|
|
|
|
VM object files.
|
|
|
|
|
|
|
|
$ make bootimage=true
|
|
|
|
$ mkdir hello
|
|
|
|
$ cd hello
|
|
|
|
$ ar x ../build/linux-i386-bootimage/libavian.a
|
|
|
|
|
|
|
|
Step 2: Create a stage1 directory and extract the contents of the
|
|
|
|
class library jar into it.
|
|
|
|
|
|
|
|
$ mkdir stage1
|
|
|
|
$ (cd stage1 && jar xf ../../build/classpath.jar)
|
|
|
|
|
|
|
|
Step 3: Build the Java code and add it to stage1.
|
|
|
|
|
|
|
|
$ cat >Hello.java <<EOF
|
|
|
|
public class Hello {
|
|
|
|
public static void main(String[] args) {
|
|
|
|
System.out.println("hello, world!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
EOF
|
|
|
|
$ javac -bootclasspath stage1 -d stage1 Hello.java
|
|
|
|
|
|
|
|
Step 4: Create a ProGuard configuration file specifying Hello.main as
|
|
|
|
the entry point.
|
|
|
|
|
|
|
|
$ cat >hello.pro <<EOF
|
|
|
|
-keep class Hello {
|
|
|
|
public static void main(java.lang.String[]);
|
|
|
|
}
|
|
|
|
EOF
|
|
|
|
|
|
|
|
Step 5: Run ProGuard with stage1 as input and stage2 as output.
|
|
|
|
|
2009-11-03 15:10:30 -07:00
|
|
|
$ java -jar ../../proguard4.4/lib/proguard.jar \
|
2009-03-15 13:49:13 -06:00
|
|
|
-injars stage1 -outjars stage2 @../vm.pro @hello.pro
|
|
|
|
|
2009-06-01 12:51:20 -06:00
|
|
|
(note: pass -dontusemixedcaseclassnames to ProGuard when building on
|
|
|
|
systems with case-insensitive filesystems such as Windows and OS X)
|
2009-03-15 13:49:13 -06:00
|
|
|
|
|
|
|
Step 6: Build the boot image.
|
|
|
|
|
2009-06-01 12:51:20 -06:00
|
|
|
$ ../build/linux-i386-bootimage/bootimage-generator stage2 bootimage.bin
|
2009-03-15 13:49:13 -06:00
|
|
|
|
|
|
|
Step 7: Make an object file out of the boot image.
|
|
|
|
|
2009-10-13 19:00:16 -06:00
|
|
|
$ ../build/${platform}-${arch}/binaryToObject \
|
|
|
|
bootimage.bin bootimage-bin.o \
|
|
|
|
_binary_bootimage_bin_start _binary_bootimage_bin_end \
|
|
|
|
${platform} ${arch} 8 writable executable
|
2009-03-15 13:49:13 -06:00
|
|
|
|
|
|
|
Step 8: Write a driver which starts the VM and runs the desired main
|
|
|
|
method. Note the bootimageBin function, which will be called by the
|
2009-04-14 09:02:13 -06:00
|
|
|
VM to get a handle to the embedded boot image. We tell the VM about
|
|
|
|
this function via the "avian.bootimage" property.
|
2009-03-15 13:49:13 -06:00
|
|
|
|
|
|
|
Note also that this example includes no resources besides class files.
|
|
|
|
If our application loaded resources such as images and properties
|
|
|
|
files via the classloader, we would also need to embed the jar file
|
|
|
|
containing them. See the previous example for instructions.
|
|
|
|
|
|
|
|
$ cat >main.cpp <<EOF
|
|
|
|
#include "stdint.h"
|
|
|
|
#include "jni.h"
|
|
|
|
|
|
|
|
#ifdef __MINGW32__
|
|
|
|
# define EXPORT __declspec(dllexport)
|
|
|
|
# define BOOTIMAGE_BIN(x) binary_bootimage_bin_##x
|
|
|
|
#else
|
|
|
|
# define EXPORT __attribute__ ((visibility("default")))
|
|
|
|
# define BOOTIMAGE_BIN(x) _binary_bootimage_bin_##x
|
|
|
|
#endif
|
|
|
|
|
|
|
|
extern "C" {
|
|
|
|
|
|
|
|
extern const uint8_t BOOTIMAGE_BIN(start)[];
|
|
|
|
extern const uint8_t BOOTIMAGE_BIN(end)[];
|
|
|
|
|
|
|
|
EXPORT const uint8_t*
|
|
|
|
bootimageBin(unsigned* size)
|
|
|
|
{
|
|
|
|
*size = BOOTIMAGE_BIN(end) - BOOTIMAGE_BIN(start);
|
|
|
|
return BOOTIMAGE_BIN(start);
|
|
|
|
}
|
|
|
|
|
|
|
|
} // extern "C"
|
|
|
|
|
|
|
|
int
|
|
|
|
main(int ac, const char** av)
|
|
|
|
{
|
|
|
|
JavaVMInitArgs vmArgs;
|
|
|
|
vmArgs.version = JNI_VERSION_1_2;
|
|
|
|
vmArgs.nOptions = 1;
|
|
|
|
vmArgs.ignoreUnrecognized = JNI_TRUE;
|
|
|
|
|
|
|
|
JavaVMOption options[vmArgs.nOptions];
|
|
|
|
vmArgs.options = options;
|
|
|
|
|
|
|
|
options[0].optionString
|
|
|
|
= const_cast<char*>("-Davian.bootimage=bootimageBin");
|
|
|
|
|
|
|
|
JavaVM* vm;
|
|
|
|
void* env;
|
|
|
|
JNI_CreateJavaVM(&vm, &env, &vmArgs);
|
|
|
|
JNIEnv* e = static_cast<JNIEnv*>(env);
|
|
|
|
|
|
|
|
jclass c = e->FindClass("Hello");
|
2009-12-16 19:25:03 -07:00
|
|
|
if (not e->ExceptionCheck()) {
|
2009-03-15 13:49:13 -06:00
|
|
|
jmethodID m = e->GetStaticMethodID(c, "main", "([Ljava/lang/String;)V");
|
2009-12-16 19:25:03 -07:00
|
|
|
if (not e->ExceptionCheck()) {
|
2009-03-15 13:49:13 -06:00
|
|
|
jclass stringClass = e->FindClass("java/lang/String");
|
2009-12-16 19:25:03 -07:00
|
|
|
if (not e->ExceptionCheck()) {
|
2009-03-15 13:49:13 -06:00
|
|
|
jobjectArray a = e->NewObjectArray(ac-1, stringClass, 0);
|
2009-12-16 19:25:03 -07:00
|
|
|
if (not e->ExceptionCheck()) {
|
2009-03-15 13:49:13 -06:00
|
|
|
for (int i = 1; i < ac; ++i) {
|
|
|
|
e->SetObjectArrayElement(a, i-1, e->NewStringUTF(av[i]));
|
|
|
|
}
|
|
|
|
|
|
|
|
e->CallStaticVoidMethod(c, m, a);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int exitCode = 0;
|
2009-12-16 19:25:03 -07:00
|
|
|
if (e->ExceptionCheck()) {
|
2009-03-15 13:49:13 -06:00
|
|
|
exitCode = -1;
|
|
|
|
e->ExceptionDescribe();
|
|
|
|
}
|
|
|
|
|
|
|
|
vm->DestroyJavaVM();
|
|
|
|
|
|
|
|
return exitCode;
|
|
|
|
}
|
|
|
|
EOF
|
|
|
|
|
|
|
|
$ g++ -I$JAVA_HOME/include -I$JAVA_HOME/include/linux \
|
|
|
|
-D_JNI_IMPLEMENTATION_ -c main.cpp -o main.o
|
|
|
|
|
|
|
|
Step 9: Link the objects produced above to produce the final
|
|
|
|
executable, and optionally strip its symbols.
|
|
|
|
|
|
|
|
$ g++ -rdynamic *.o -ldl -lpthread -lz -o hello
|
|
|
|
$ strip --strip-all hello
|