anvil
A build tool for Haxe native extensions built using ammer
What it does
anvil
will allow users of a native library extension to build normally without having to perform native build maintenance tasks (like manually building the native code, or moving the binaries to projects that use it).
It provides a single class that provides a single static entry method for library authors to integrate their C/C++ build infrastructure into their haxelib.
In short, it will make sure all of your native library source files and binaries end up in the right place to be consumed by a user of your library and will attempt to build the source in the users project (hence the binaries end up in the right place)
All you need to do is include an initialization macro and an .anvilrc
config file (JSON) at the root of your libraries project (and an ammer ready library), and all the user needs to do is include your library (and possibly specify other non-library specific config).
Not only will building projects with your native extensions build the native code (using your build command), but the Haxe compilation server may do so as well, allowing the IDE to offload native library building to a background service.
TODO
Test MinGW Test GC/Linux/OSX
How it works
anvil
provides a simple build script to simulate a cohesive Haxe/native extension build for a given native library.
A library author simply needs to reference the Anvil library via -lib anvil
and create an initialization macro like so:
class SimpleAnvilBootstrap {
public static function run() {
anvil.Anvil.run();
}
}
Then in the library's extraParams.hxml
:
-D ammer.lib.<ammerLib>.library=<ammerLibPath>
-D ammer.lib.<ammerLib>.include=<ammerIncludePath>
--macro SimpleAnvilBootstrap.run()
anvil
determines the root directory of a haxe library (from the source file on the user's computer) by looking in parent directories for an .anvilrc
file.
The .anvilrc
is just a JSON representation of this Haxe typedef:
typedef AnvilConfig = {
var ?windows:Array<AnvilPlatformConfig>;
var ?linux:Array<AnvilPlatformConfig>;
var ?bsd:Array<AnvilPlatformConfig>;
var ?mac:Array<AnvilPlatformConfig>;
}
typedef AnvilPlatformConfig = {
var ammerLib:String; // e.g. what you used to define -D ammer.lib.<ammerLib>.library etc...
var nativePath:String; // the path to the native code from the root of your project
var ?hxMakefile:String; // ** NEW ** build your project using an HxMakefile; this assumes the user has HxMake.exe in their PATH
var ?buildCmd:String; // the build command to run (this will run with the native path as its working directory; ensure all environment variables are set in order for this command to be successful)
var ?outputBinaries:Array<String>; // list of binaries that need to be processed; if this isn't provided, anvil will infer what binaries to move
// if anvil infers this, it will assume that if any library binary exists, the project was already fully built
// otherwise it will compare the directory contents to the outputBinaries to determine if the project is already built
// when determining whether or not to skip compilation
// i.e. If using MSVC toolchain, vsdevcmd.bat/vsvars32.bat need to be run in the shell before this.
var ?buildArgs:Array<String>; // arguments for the build command. This works like Sys.command or new sys.io.Process, can be omitted (with args in cmd name)
var ?verbose:Bool; // pipes the stdout of the build command to the stdout of the haxe build command.
var ?libPath:String; // where the binaries are, if not at nativePath
var ?includePath:String; // where the headers are if not at nativePath
var ?deployInfo:{p
var type:DeployType; // with-user-output or to-path
var ?dest:String; // if to-path, the destination path to put built binaries
};
var ?cacheMode:CacheMode; // cache mode, either always, on-change, or never.
// on-change will rebuild whenever the source files are changed
// this is the default setting
}
If you specify hxMakefile
, do not specify buildCmd
(the command to run your hxMakefile is the build command), and vice-versa, do not specify hxMakefile
if you want to run a command at the command line instead of using HxMake
.
You can read about HxMakefiles
here
For example, say you create a native-crypto
library that has different native implementations, you can still use the same Haxe code, just specify different nativePath
s:
{
"windows": [
{
"ammerLib": "native-crypto",
"nativePath": "native-win",
"hxMakefile": "HxMakefile.win"
}
],
"linux": [
{
"ammerLib": "native-crypto",
"nativePath": "native-linux",
"hxMakefile": "HxMakefile.linux"
}
],
"mac": [
{
"ammerLib": "native-crypto",
"nativePath": "native-mac",
"hxMakefile": "HxMakefile.osx"
}
]
}
Library Authors
Authors can also use this tool for testing, simply call the same initialization macro you put in extraParams.hxml
in whatever.hxml file builds/runs your tests.
Library Users
You can tell anvil
where your output binaries will be so it can also move the required library binaries to your Haxe build output directory by using -D anvil.output=<path>
Example
Because the test is located in the same repo as anvil
, it is hard to tell what anvil
really does.
Use Case:
-
a user will have a Haxe project probably somewhere, say
C:/User/Documents/projects
, and it contains anhxml
(say,C:/User/Documents/projects/build.hxml
) -
your library will be located at something like
C:/Users/AppData/Roaming/Haxe/Haxelib/my_native_haxelib_extension/1,0,0,
-
The user includes your haxe library, and its
extraParams.hxml
(where you put the Bootsrap code at) -
The user then runs
haxe build.hxml
atC:/User/Documents/projects
to build their project including your library -
The Bootstrap kicks of
anvil
: -
anvil
will then copy thenativePath
(saynative
) from theC:/Users/AppData/Roaming/Haxe/Haxelib/my_native_haxelib_extension/1,0,0,/native
directory toC:/User/Documents/projects/native_extensions/$ammerLib
(say$ammerLib
ismy_native_lib
) -
Set the CWD to
C:/User/Documents/projects/native_extensions/my_native_lib
-
Set the compiler flags
-D ammer.lib.<ammerLib>.library
and-D ammer.lib.<ammerLib>.include
based onlibPath
andincludePath
relative toC:/User/Documents/projects/native_extensions/my_native_lib
(this should allow the user's project to build smoothly; bothlibPath
andincludePath
may be blank, in which case they arenativePath
) -
Run
$buildCmd
-
Check for library binaries (
dll
,dylib
,so
) in the folder indicated byammer.lib.<ammerLib>.library
(i.e.:C:/User/Documents/projects/native_extensions/my_native_lib
, assuming nolibPath
is set) -
Move them to the output folder specified by the user of your library (and other Ammer native extension libraries) with
-D anvil.output=where/user/puts/their/bins
-
Set CWD back to the original running directory
-
Continue regular build process (at this point, your project should be able to be properly linked and utilized by the user)