This code attempts to implement some of the ideas mentioned at http://bugs.sun.com/view_bug.do?bug_id=5049299.
Basically, the goal is to avoid a fork() call whenever we need to spawn a process so that a large project does not have to unnecessarily duplicate its virtual memory usage.
If you read the Theory of Operation, you'll learn the trick. The upshot is that this library requires three discrete components to work correctly:
- The JAR (
jlinuxfork.jar
) - The dynamic-linked library (
libjlinuxfork-{arch}.so
) - The binary redirect program (
binrunner
)
While the JAR file is standard Java and can be dropped in a dependency management via pom or other systems, the other two components make the usage a of this library a little bit trickier.
The simplest install is to just install the components to the operating system. This requires no configuration at run time and can be built thus:
$ git clone clone https://[email protected]/axiak/java_posix_spawn.git $ cd java_posix_spawn $ ant $ cd target $ sudo cp libjlinuxfork*so /usr/lib/ $ sudo cp binrunner /usr/bin $ cp jlinuxfork.jar YOUR_PROJECT_LIB
To run the test to see that it works, do the following:
$ git clone clone https://[email protected]/axiak/java_posix_spawn.git $ cd java_posix_spawn $ ant $ cd test $ make
This should then print out the output of running "ls -l /" without any of the duplication of virtual address space.
The question now is, is it behaving better than the standard java library? To be sure, follow these steps:
- Start a VM that does not have much memory (e.g. 512MB)
- Edit test/
Makefile
to adjustXms
andXmx
to make the heap large enough. - In the
test
directory, runmake
to see this library, and runmake runfallback
to see the fallback action.
If you don't want to place the libraries in system-level directories, then there are two java settings you can define when running your java application:
java.system.path
- Set to the directory containing the.so
file.posixspawn.binrunner
- Set to the complete path of yourbinrunner
file.
For example, to run the TestClass
in the repository, the bundled Makefile
, runs the following:
java -classpath ../target/jlinuxfork.jar:. -Djava.library.path=../target -Dposixspawn.binrunner=../target/binrunner TestClass
In linux, a fork
causes linux to duplicate the allocation of virtual memory
in the virtual memory address space. This works well for lots of small applications spawning each other, but in the java world large applications can spawn small applications in rapid succession and leave the operating system unable to manage the virtual address space.
The workaround is to use vfork
rather than fork
before spawning another process. With fork, this is how java spawns a process:
- Create 3 pipes (for stdin, stdout, and stderr of the child process)
- Fork the process
- Redirect stdin, stdout, and stderr to the appropriate pipes
- Run
execve
or some equivalent to run a different application (andbrk
)
In order for vfork
to work in a thread-safe manner, one cannot reference memory
space in the parent process after the vfork
call. (The only legal calls are _exit
and execve
. It is not legal to redirect file descriptors after a call to vfork, since we haven't created the new memory address space to redirect yet.) So all we can do in vfork
is run an external program.
The trick to get around this limitation of vfork
is to create an intermediary program, called binrunner
. binrunner is a really simple C program that expects the following:
Usage: binrunner stdin# stdout# stderr# chdir program [argv0 ... ]
So essentially you tell it what file descriptors you want redirected, what directory you want to change into, what program you want to run, and you complete running arguments at the end to pass to the next program. Binrunner will then perform the pipe redirection, call chdir() and then call execvp.
So this is how this library will spawn a new process:
- Create 3 pipes
- Call vfork
- Run binrunner with the pipe information, the new directory, and the program information
- binrunner redirects the pipes
- binrunner calls
chdir()
- binrunner calls
execvp
to run the application (andbrk
)
The library is actually slightly misnamed -- rather than using posix_spawn
blindly, it actually calls vfork
manually. This is due to that fact that posix_spawn does not produce a useful error in the case where the execve
fails. By calling vfork
manually, the application can safely write an error
to stderr
rather than keeping completely silent.
Beyond this README, I put up a shell of the API at http://axiak.github.com/java_posix_spawn/
The API is supposed to mirror that of Java's Runtime, so it shouldn't be anything new. The only
added method is SpawnRuntime.isLinuxSpawnLoaded()
, which indicates if the java library was
successfully able to load the dynamic library. On Windows you would expect that method to always
return false.
- Java >= 1.4
- A C compiler
- make
- ant (>=1.7?)
The API is written such that it will fall back to the standard java Runtime API if it cannot load the dynamic libraries. This means that windows can just run the java code without any support for compilation (since its Runtime exec doesn't suffer from the but, it's safe).
As for non-windows systems, this library was tested on linux 32- and 64-bit. No testing has been done on other posix-compliant systems, but the code strictly adheres to posix standards.
The library is released in the Modified BSD License. See LICENSE for more detail.
None at the moment. Please file an issue if you find any.