<?xml version="1.0"?>

<project name="COOJA Simulator - JNI Tests" default="about" basedir=".">
	<property environment="env"/>

	<target name="help">
		<echo>
The majority of problems encountered during these tests are related to 
the compilation and linking stage of the Contiki libraries.
Below follows a number of configurations that work for different users.

Linux users with the GNU toolchain may try:
	PATH_C_COMPILER = gcc
	COMPILER_ARGS = 
  LINK_COMMAND_1 = gcc -shared -Wl,-Map=$(MAPFILE) -o $(LIBFILE)
  LINK_COMMAND_2 = 
  PARSE_WITH_COMMAND=false
			
Windows users with recent Cygwin may try:
	PATH_C_COMPILER = gcc
	COMPILER_ARGS = -mno-cygwin -Wall -I'$(JAVA_HOME)/include' -I'$(JAVA_HOME)/include/win32'
	LINK_COMMAND_1 = gcc -mno-cygwin -shared -Wl,-Map=$(MAPFILE) -Wl,--add-stdcall-alias -o $(LIBFILE)
	LINK_COMMAND_2 = -L/usr/lib/mingw -lmsvcrt

Windows users with older Cygwin may try:
	PATH_C_COMPILER = gcc
	COMPILER_ARGS = -mno-cygwin -I'$(JAVA_HOME)/include' -I'$(JAVA_HOME)/include/win32'
	LINK_COMMAND_1 = ld -Map=$(MAPFILE) -shared --add-stdcall-alias /usr/lib/mingw/dllcrt2.o -o $(LIBFILE)
	LINK_COMMAND_2 = -L/usr/lib/mingw -lmingw32 -lmingwex -lmsvcrt

Windows users with recent Cygwin and separate MinGW tools may try:
In recent (early 2007) MinGW-Cygwin versions, the JNI support has been limited.
This may cause COOJA not to be able to load libraries generated using gcc's -mno-cygwin flag.
One possible solution is to download "real" MinGW (http://www.mingw.org), and install it separately
from Cygwin (for example in c:\mingw). Try using the following settings:
	PATH_C_COMPILER = gcc
	COMPILER_ARGS = -Wall -D_JNI_IMPLEMENTATION_ -I'$(JAVA_HOME)/include' -I'$(JAVA_HOME)/include/win32'
	LINK_COMMAND_1 = gcc -shared -Wl,-Map=$(MAPFILE) -Wall -D_JNI_IMPLEMENTATION_ -Wl,--kill-at -o $(LIBFILE)
	LINK_COMMAND_2 = 

Mac users may try (see Mac note below):
	PATH_C_COMPILER = gcc
	PARSE_WITH_COMMAND = true
	PARSE_COMMAND = nmandsize $(LIBFILE)
	COMPILER_ARGS = -Wall -I/System/Library/Frameworks/JavaVM.framework/Versions/CurrentJDK/Headers -dynamiclib -fno-common
	LINK_COMMAND_1 = gcc -dynamiclib -fno-common -o $(LIBFILE)
	LINK_COMMAND_2 = -framework JavaVM

Mac note:
	The Mac OS X support is experimental, and currently depends on an external script to work.
	The script 'nmandsize' is available in the subdirectory 'mac_users' and depends on both 'nm' and 'size'.
	The purpose of the script is to gather information which COOJA uses to determine the memory sections addresses.
	Observe that, if used, this script must be in the path.
		</echo>
	</target>
	
	<target name="about">
		<echo>
The COOJA Simulator - JNI Tests
-------------------------------

These tests can be used to help understand COOJA errors, and to configure COOJA for new users.
For COOJA to compile JNI libraries successfully, tests 2-5 must be completed.
In level3, only level3a or level3b is necessary to pass.

You may have to change the configuration in the file 'exttools.cfg'.
When all tests pass, these settings should be entered into the COOJA External tool settings dialog.

Before running the tests, the COOJA JAR file must be created:
> ant compile_cooja
			
To run the first test:
> ant level2

For more information including configuration examples:
> ant help

> ant level2
	[compilation test]
	Compiles level2.c to level2.library, using both c compiler and linker.
	Java class loads the library and calls a simple native function.

> ant level3a
	[address parsing using map file]

> ant level3b
	[address parsing using command]

> ant level4
	[fetching reference var]
	Calculates offset between relative and absolute memory.
	A simple native function increases two counters (from both data and bss sections).

> ant level5
	[fetches and restores memory segments - the final test]
	A simple native function increases two counters (from both data and bss sections).
	The current memory (data+bss sections) is fetched and restored between function calls.
	The counters should be restored with the memory!
	</echo>	
	</target>

	<target name="clean">
		<delete>
			 <fileset dir="." includes="**/*.class" />
			 <fileset dir="." includes="**/*.library" />
			 <fileset dir="." includes="**/*.o" />
			 <fileset dir="." includes="**/*.map" />
			 <fileset dir="." includes="**/*.log" />
			 <fileset dir="." includes="build.temp" />
		 </delete>
	</target>

	<target name="init">
		<tstamp/>
	</target>

	<target name="compile_library" depends="init">
		<property name="SRCFILE" value="${LEVEL}.c"/>
		<property name="OBJFILE" value="${LEVEL}.o"/>
		<property name="LIBFILE" value="${LEVEL}.library"/>
		<property name="MAPFILE" value="${LEVEL}.map"/>
		<property name="ARFILE" value="${LEVEL}.a"/>

		<property file="exttools.cfg"/>
		<property name="COMPILE_COMMAND" value="${PATH_C_COMPILER} ${COMPILER_ARGS} -c ${SRCFILE} -o ${OBJFILE}"/>
		<property name="LINKER_COMMAND" value="${LINK_COMMAND_1} ${OBJFILE} ${LINK_COMMAND_2}"/>

		<propertyfile file="build.temp">
			<entry key="COMPILE_EXEC" value="${COMPILE_COMMAND}"/>
			<entry key="COMPILE_ARGS" value="${COMPILE_COMMAND}"/>
			<entry key="LINKER_EXEC" value="${LINKER_COMMAND}"/>
			<entry key="LINKER_ARGS" value="${LINKER_COMMAND}"/>
		</propertyfile>
		<replace file="build.temp" token="$(JAVA_HOME)" value="${env.JAVA_HOME}"/>
		<replace file="build.temp" token="$(LIBFILE)" value="${LIBFILE}"/>
		<replace file="build.temp" token="$(MAPFILE)" value="${MAPFILE}"/>
	 	<replace file="build.temp" token="$(ARFILE)" value="${ARFILE}"/>
		<replaceregexp 
			file="build.temp" match="COMPILE_EXEC=(\w*).*" 
			replace="COMPILE_EXEC=\1" byline="true"/>
		<replaceregexp 
			file="build.temp" match="COMPILE_ARGS=\w*(.*)" 
			replace="COMPILE_ARGS=\1" byline="true"/>
		<replaceregexp 
			file="build.temp" match="LINKER_EXEC=(\w*).*" 
			replace="LINKER_EXEC=\1" byline="true"/>
		<replaceregexp 
			file="build.temp" match="LINKER_ARGS=\w*(.*)" 
			replace="LINKER_ARGS=\1" byline="true"/>
		<property file="build.temp"/>
		
		<echo message="EXECUTING: ${COMPILE_EXEC} ${COMPILE_ARGS}"/>
		<exec dir="${LEVEL}" executable="${COMPILE_EXEC}" failonerror="true">
			<arg line="${COMPILE_ARGS}"/>
		</exec>
		
		<echo message="EXECUTING: ${LINKER_EXEC} ${LINKER_ARGS}"/>
		<exec dir="${LEVEL}" executable="${LINKER_EXEC}" failonerror="true">
			<arg line="${LINKER_ARGS}"/>
		</exec>
	</target>

	<target name="compile_cooja" depends="init">
		<ant dir="../../" antfile="build.xml" target="jar"/>
	</target>
	
	<target name="level2" depends="init">
		<property name="LEVEL" value="level2"/>
		<antcall target="compile_library" inheritall="true"/>
		
		<javac srcdir="${LEVEL}" destdir="${LEVEL}"/>
		<java fork="yes" dir="${LEVEL}" classname="Level2"/>
	</target>

	<target name="level3">
		<echo>
Run either
	> ant level3a
or
	> ant level3b			

3a tests parsing library addresses using the map file generated at link time.
3b tests parsing library addresses using a command.
At least one of the tests must pass.

After passing, the 'exttools.cfg' must be updated.
The external tools setting:
	PARSE_WITH_COMMAND=true
causes COOJA (and the following tests) to use the command to parse library addresses,
and should be used if test 3b passed.

	PARSE_WITH_COMMAND=false
causes COOJA to parse the generated map file, and should be used if test 3a passed.
			
		</echo>
	</target>

	<target name="level3a" depends="init">
		<property name="LEVEL" value="level3a"/>
		<antcall target="compile_library" inheritall="true"/>
		
		<javac srcdir="${LEVEL}" destdir="${LEVEL}" classpath="../../dist/cooja.jar"/>
		<java fork="yes" dir="${LEVEL}" classname="Level3a">
			<classpath>
				<pathelement location="../../dist/cooja.jar"/>
				<pathelement location="${LEVEL}"/>
			</classpath>
		</java>	
	</target>

	<target name="level3b" depends="init">
		<property name="LEVEL" value="level3b"/>
		<antcall target="compile_library" inheritall="true"/>
		
		<javac srcdir="${LEVEL}" destdir="${LEVEL}" classpath="../../dist/cooja.jar"/>
		<java fork="yes" dir="${LEVEL}" classname="Level3b">
			<classpath>
				<pathelement location="../../dist/cooja.jar"/>
				<pathelement location="${LEVEL}"/>
			</classpath>
		</java>	
	</target>

	<target name="level4" depends="init">
		<property name="LEVEL" value="level4"/>
		<antcall target="compile_library" inheritall="true"/>
		
		<javac srcdir="${LEVEL}" destdir="${LEVEL}" classpath="../../dist/cooja.jar"/>
		<java fork="yes" dir="${LEVEL}" classname="Level4">
			<classpath>
				<pathelement location="../../dist/cooja.jar"/>
				<pathelement location="${LEVEL}"/>
			</classpath>
		</java>	
	</target>

	<target name="level5" depends="init">
		<property name="LEVEL" value="level5"/>
		<antcall target="compile_library" inheritall="true"/>
		
		<javac srcdir="${LEVEL}" destdir="${LEVEL}" classpath="../../dist/cooja.jar"/>
		<java fork="yes" dir="${LEVEL}" classname="Level5">
			<classpath>
				<pathelement location="../../dist/cooja.jar"/>
				<pathelement location="${LEVEL}"/>
			</classpath>
		</java>	
	</target>

</project>