Introduction
This article explains how can we create a Java Virtual Machine Launcher (like java.exe or javaw.exe). It explores how the Java Virtual Machine launches a Java application. It gives you more ideas on the JDK or JRE you are using. This launcher is very useful in Cygwin (Linux emulator) with Java Native Interface. This article assumes a basic understanding of JNI.
How to Create (Launch) JVM ?
The standard launcher command (java
or javaw.exe) in JDK or JRE is no more than a simple C program linked with the Java Virtual Machine. The launcher parses the command line arguments, loads the virtual machine, and runs Java applications through the invocation interface.
Before going further, let us discuss some of the structures and functions used to create the JVM.
JavaVMInitArgs
The arguments to the Java Virtual Machine are defined in jni.h as follows:
typedef struct JavaVMInitArgs {
jint version;
jint nOptions;
JavaVMOption *options;
jboolean ignoreUnrecognized;
} JavaVMInitArgs;
typedef struct JavaVMOption {
char *optionString;
void *extraInfo;
} JavaVMOption;
JNI_CreateJavaVM
jint JNI_CreateJavaVM(JavaVM **p_vm, JNIEnv **p_env, void *vm_args);
The first argument is a pointer to a JavaVM
pointer. The JavaVM
structure can be used to attach and detach native threads to/from the virtual machine. The second argument is a pointer to a JNIEnv
pointer. A JNIEnv
structure is the main workhorse for JNI programming. It roughly corresponds to a particular Java thread. The JNIEnv
returned from JNI_CreateJavaVM()
, thus, represents the VM's main thread. The third argument is a pointer to an arbitrary pointer, and consists of the VM arguments. Here is the example code that creates a virtual machine instance:
JavaVMInitArgs vm_args;
JavaVMOption options[4];
options[0].optionString ="-Djava.compiler=NONE";
options[1].optionString = "-Djava.class.path=c:\\myclasses";
options[2].optionString = "-Djava.library.path=c:\\mylibs";
options[3].optionString = "-verbose:jni";
vm_args.version = JNI_VERSION_1_2;
vm_args.options = options;
vm_args.nOptions = 4;
vm_args.ignoreUnrecognized = TRUE;
typedef jint (JNICALL CreateJavaVM_t)(JavaVM **pvm, void **env, void *args);
HINSTANCE hinst = LoadLibrary("jvm.dll")
CreateJavaVM_t *pfnCreateJavaVM = GetProcAddress(hinst, "JNI_CreateJavaVM");
Int iRetval = pfnCreateJavaVM((&vm, (void **)&env,
&vm_args);
if (res < 0) {
...
Invoking the Class
To launch any Java program, first we have to find out the class we specified and then we can call the main method of that class. We can pass arguments also to the Java program we are launching. The following piece of code illustrates this:
jclass jcJclass = psJNIEnv->FindClass(mainClassName);
jmethodID jmMainMethod =
psJNIEnv-> GetStaticMethodID(jcJclass, "main", "([Ljava/lang/String;)V");
psJNIEnv->CallStaticVoidMethod(jcJclass, jmMainMethod, joApplicationArgs);
Where Can I Use this Launcher?
Have you ever tried to load the libraries which are compiled under cygwin from Java? You can see that your application hangs during System.loadLibrary()
. The problem is, the library built under cygwin depends on cygwin1.dll. If there's no other cygwin-using process running, then dynamically loading a DLL which depends (at load-time) on cygwin causes the process to deadlock. So we need a cygwin process to launch the JVM.
About Cygwin
Cygwin is a Linux-like environment for Windows. It consists of two parts:
- A DLL (cygwin1.dll) which acts as a Linux API emulation layer providing substantial Linux API functionality.
- A collection of tools which provide Linux look and feel. Using Cygwin, programs written for compilation under Linux can often be cross-compiled for Win32 with minimal changes. It also helps to work in Windows with the Linux libraries.
Using the Code
The code for this launcher is written mainly to work with eclipse. Now it handles only basic JVM arguments.
How to Compile the Code
->Open a cygwin shell ->Run the compiler. (g++ cvm.cpp -o javaw
)
How to Use the Launcher
Note: This launcher may not work with all JRE/JDK versions, because, the arguments to the JVM differ from version to version. I tested with j2re1.4.2_06 and it works.