Version 0.20, March 04, 2017
Albert Gräf <aggraef@gmail.com>
pure-gen – Pure interface generator
pure-gen is a C interface generator for the Pure language. It takes a C header file as input and generates a corresponding Pure module with the constant definitions and extern declarations needed to use the C module from Pure. pure-gen can also generate FFI interfaces rather than externs (using the pure-ffi module), and it can optionally create a C wrapper module which allows you to create interfaces to pretty much any code which can be called via C.
Interface type (extern, c, ffi or c-ffi). Default is extern. The extern and c types generate Pure extern declarations, which is what you want in most cases. ffi and c-ffi employ Pure’s libffi interface instead. The c and c-ffi types cause an additional C wrapper module to be created (see Generating C Code). These can also be combined with the -auto suffix which creates C wrappers only when needed to get C struct arguments and returns working, see Dealing with C Structs for details.
Add dynamic library module to be imported in the Pure output file. Default is -l c-file (the filename specified with -c, see below, without filename extension) if one of the -fc options was specified, none otherwise.
Prefix to be prepended to C wrapper symbols (-fc and friends). Default is Pure_.
Include “hidden” symbols in the output. Built-in preprocessor symbols and symbols starting with an underscore are excluded unless this option is specified.
Selection of C symbols to be included in the output. pattern takes the form [glob-patterns::][regex-pattern], designating a comma separated list of glob patterns matching the source filenames, and an extended regular expression matching the symbols to be processed. See glob(7) and regex(7). The default pattern is empty which matches all symbols in all source modules.
Specify a C template file to be used with C wrapper generation (-fc). See Generating C Code for details.
Specify an alternate C template file to be used with C wrapper generation (-fc). See Generating C Code for details.
pure-gen generates Pure bindings for C functions from a C header file. For instance, the command
pure-gen foo.h
creates a Pure module foo.pure with extern declarations for the constants (#defines and enums) and C routines declared in the given C header file and (recursively) its includes.
pure-gen only accepts a single header file on the command line. If you need to parse more than one header in a single run, you can just create a dummy header with all the necessary #includes in it and pass that to pure-gen instead.
When invoked with the -n option, pure-gen performs a dry run in which it only parses the input without actually generating any output files. This is useful for checking the input (possibly in combination with the -e, -v and/or -w options) before generating output. A particularly useful example is
pure-gen -ne foo.h \
| awk '$1=="#" && $2~/^[0-9]+$/ && $3!~/^"<.*>"$/ { print $3 }' \
| sort | uniq
which prints on standard output all headers which are included in the source. This helps to decide which headers you want to be included in the output, so that you can set up a corresponding filter patterns (-s and -x options, see below).
The -I, -D and -U options are simply passed to the C preprocessor, as well as any other option or argument escaped with the -C flag. This is handy if you need to define additional preprocessor symbols, add directories to the include search path, etc., see cpp(1) for details.
There are some other options which affect the generated output. In particular, -f c generates a C wrapper module along with the Pure module (see Generating C Code below), and -f ffi generates a wrapper using Pure’s ffi module. Moreover, -l libfoo generates a using "lib:libfoo" declaration in the Pure source, for modules which require a shared library to be loaded. Any number of -l options can be specified.
Other options for more advanced uses are explained in the following sections.
Note that pure-gen always parses the given header file as well as all its includes. If the header file includes system headers, by default you will get those declarations as well. This is often undesirable. As a remedy, pure-gen normally excludes built-in #defines of the C preprocessor, as well as identifiers with a leading underscore (which are often found in system headers) from processing. You can use the -a option to disable this, so that all these symbols are included as well.
In addition, the -s and -x options enable you to filter C symbols using the source filename and the symbol as search criteria. For instance, to just generate code for a single header foo.h and none of the other headers included in foo.h, you can invoke pure-gen as follows:
pure-gen -s foo.h:: foo.h
Note that even in this case all included headers will be parsed so that #defined constants and enum values can be resolved, but the generated output will only contain definitions and declarations from the given header file.
In general, the -s option takes an argument of the form glob-patterns::regex-pattern denoting a comma-separated list of glob patterns to be matched against the source filename in which the symbol resides, and an extended regex to be matched against the symbol itself. The glob-patterns:: part can also be omitted in which case it defaults to :: which matches any source file. The regex can also be empty, in which case it matches any symbol. The generated output will contain only the constant and function symbols matching the given regex, from source files matching any of the the glob patterns. Thus, for instance, the option -s foo.h,bar.h::^(foo|bar)_ pulls all symbols prefixed with either foo_ or bar_ from the files foo.h and bar.h in the current directory.
Instead of :: you can also use a single semicolon ; to separate glob and regex pattern. This is mainly for Windows compatibility, where the msys shell sometimes eats the colons or changes them to ;.
The -x option works exactly the same, but excludes all matching symbols from the selection. Thus, e.g., the option -x ^bar_ causes all symbols with the prefix bar_ to not be included in the output module.
Processing of glob patterns is performed using the customary rules for filename matching, see glob(7) for details. Note that some include files may be specified using a full pathname. This is the case, in particular, for system includes such as #include <stdio.h>, which are resolved by the C preprocessor employing a search of the system include directories (as well as any directories named with the -I option).
Since the * and ? wildcards never match the pathname separator /, you have to specify the path in the glob patterns in such cases. Thus, e.g., if the foo.h file actually lives in either /usr/include or /usr/local/include, then it must be matched using a pattern like /usr/include/*.h,/usr/local/include/*.h::. Just foo.h:: will not work in this case. On the other hand, if you have set up your C sources in some local directory then specifying a relative pathname is ok.
The -s option is often used in conjuction with the -p option, which lets you specify a “module name prefix” which should be stripped off from C symbols. Case is insignificant and a trailing underscore will be removed as well, so -p foo turns fooBar into Bar and FOO_BAR into BAR. Moreover, the -m option allows you to specify the name of a Pure namespace in which the resulting constants and functions are to be declared. So, for instance, -s "^(foo|FOO)" -p foo -m foo will select all symbols starting with the foo or FOO prefix, stripping the prefix from the selected symbols and finally adding a foo:: namespace qualifier to them instead.
As already mentioned, pure-gen can be invoked with the -fc or -fc-ffi option to create a C wrapper module along with the Pure module it generates. There are various situations in which this is preferable, e.g.:
The latter case might arise, e.g., if the module uses non-C linkage or calling conventions, or if some of the operations to be wrapped are actually implemented as C macros. (Note that in order to wrap macros as functions you’ll have to create a staged header which declares the macros as C functions, so that they are wrapped in the C module. pure-gen doesn’t do this automatically.)
Another important case is that some of the C routines pass C structs by value or return them as results. This is discussed in more detail in the following section.
For instance, let’s say that we want to generate a wrapper foo.c from the foo.h header file whose operations are implemented in some library libfoo.a or libfoo.so. A command like the following generates both the C wrapper and the corresponding Pure module:
pure-gen -fc foo.h
This creates foo.pure and foo.c, with an import clause for "lib:foo" at the beginning of the Pure module. (You can also change the name of the Pure and C output files using the -o and -c options, respectively.)
The generated wrapper is just an ordinary C file which should be compiled to a shared object (dll on Windows) as usual. E.g., using gcc on Linux:
gcc -shared -o foo.so foo.c -lfoo
That’s all. You should now be able to use the foo module by just putting the declaration using foo; into your programs. The same approach also works with the ffi interface if you replace the -fc option with -fc-ffi.
You can also adjust the C wrapper code to some extent by providing your own template file, which has the following format:
/* frontmatter here */
#include %h
%%
/* wrapper here */
%r %w(%p)
{
return %n(%a);
}
Note that the code up to the symbol %% on a line by itself denotes “frontmatter” which gets inserted at the beginning of the C file. (The frontmatter section can also be empty or missing altogether if you don’t need it, but usually it will contain at least an #include for the input header file.)
The rest of the template is the code for each wrapper function. Substitutions of various syntactical fragments of the function definition is performed using the following placeholders:
%h input header file
%r return type of the function
%w the name of the wrapper function
%p declaration of the formal parameters of the wrapper function
%n the real function name (i.e., the name of the target C function to be called)
%a the arguments of the function call (formal parameters with types stripped off)
%% escapes a literal %
A default template is provided if you don’t specify one (which looks pretty much like the template above, minus the comments). A custom template is specified with the -t option. (There’s also a -T option to specify an “alternate” template for dealing with routines returning struct values, see Dealing with C Structs.)
For instance, suppose that we place the sample template above into a file foo.templ and invoke pure-gen on the foo.h header file as follows:
pure-gen -fc -t foo.templ foo.h
Then in foo.c you’d get C output code like the following:
/* frontmatter here */
#include "foo.h"
/* wrapper here */
void Pure_foo(int arg0, void* arg1)
{
return foo(arg0, arg1);
}
/* wrapper here */
int Pure_bar(int arg0)
{
return bar(arg0);
}
As indicated, the wrapper function names are usually stropped with the Pure_ prefix. You can change this with the -P option.
This also works great to create boilerplate code for new modules. For this purpose the following template will do the trick:
/* Add #includes etc. here. */
%%
%r %n(%p)
{
/* Enter code of %n here. */
}
Modern C compilers allow you to pass C structs by value or return them as results from a C function. This represents a problem, because Pure doesn’t provide any support for that in its extern declarations. Even Pure’s libffi interface only has limited support for C structs (no unions, no bit fields), and at present pure-gen itself does not keep track of the internal structure of C structs either.
Hence pure-gen will bark if you try to wrap an operation which passes or returns a C struct, printing a warning message like the following which indicates that the given function could not be wrapped:
Warning: foo: struct argument or return type, try -fc-auto
What Pure does know is how to pass and return pointers to C structs in its C interface. This makes it possible to deal with struct arguments and return values in the C wrapper. To make this work, you need to create a C wrapper module as explained in the previous section. However, as C wrappers are only needed for functions which actually have struct arguments or return values, you can also use the -fc-auto option (or -fc-ffi-auto if you prefer the ffi interface) to only generate the C wrapper when required. This saves the overhead of an extra function call if it’s not actually needed.
Struct arguments in the original C function then become struct pointers in the wrapper function. E.g., if the function is declared in the header as follows:
typedef struct { double x, y; } point;
extern double foo(point p);
Then the generated wrapper code becomes:
double Pure_foo(point* arg0)
{
return foo(*arg0);
}
Which is declared in the Pure interface as:
extern double Pure_foo(point*) = foo;
Struct return values are handled by returning a pointer to a static variable holding the return value. E.g.,
extern point bar(double x, double y);
becomes:
point* Pure_bar(double arg0, double arg1)
{
static point ret;
ret = bar(arg0, arg1); return &ret;
}
Which is declared in the Pure interface as:
extern point* Pure_bar(double, double) = bar;
(Note that the generated code in this case comes from an alternate template. It’s possible to configure the alternate template just like the normal one, using the -T option instead of -t. See the Generating C Code section above for details about code templates.)
In a Pure script you can now call foo and bar as:
> foo (bar 0.0 1.0);
Note, however, that the pointer returned by bar points to static storage which will be overwritten each time you invoke the bar function. Thus in the following example both u and v will point to the same point struct, namely that defined by the latter call to bar:
> let u = bar 1.0 0.0; let v = bar 0.0 1.0;
Which most likely is not what you want. To avoid this, you’ll have to take dynamic copies of returned structs. It’s possible to do this manually by fiddling around with malloc and memcpy, but the most convenient way is to employ the struct functions provided by Pure’s ffi module:
> using ffi;
> let point_t = struct_t (double_t, double_t);
> let u = copy_struct point_t (bar 1.0 0.0);
> let v = copy_struct point_t (bar 0.0 1.0);
Now u and v point to different, malloc’d structs which even take care of freeing themselves when they are no longer needed. Moreover, the ffi module also allows you to access the members of the structs in a direct fashion. Please refer to the pure-ffi documentation for further details.
pure-gen currently requires gcc (-E) as the C preprocessor. It also needs a version of gcc which understands the -fdirectives-only option, which means gcc 4.3 or later. It will run with older versions of gcc, but then you’ll get an error message from gcc indicating that it doesn’t understand the -fdirectives-only option. pure-gen then won’t be able to extract any #defined constants from the header files.
pure-gen itself is written in Pure, but uses a C parser implemented in Haskell, based on the Language.C library written by Manuel Chakravarty and others.
pure-gen can only generate C bindings at this time. Other languages may have their own calling conventions which make it hard or even impossible to call them directly through Pure’s extern interface. However, if your C compiler knows how to call the other language, then it may be possible to interface to modules written in that language by faking a C header for the module and generating a C wrapper with a custom code template, as described in Generating C Code. In principle, this approach should even work with behemoths like C++, although it might be easier to use third-party tools like SWIG for that purpose.
In difference to SWIG and similar tools, pure-gen doesn’t require you to write any special “interface files”, is controlled entirely by command line options, and the amount of marshalling overhead in C wrappers is negligible. This is possible since pure-gen targets only the Pure-C interface and Pure has good support for interfacing to C built into the language already.
pure-gen usually works pretty well if the processed header files are written in a fairly clean fashion. Nevertheless, some libraries defy fully automatic wrapper generation and may thus require staged headers and/or manual editing of the generated output to get a nice wrapper module.
In complex cases it may also be necessary to assemble the output of several runs of pure-gen for different combinations of header files, symbol selections and/or namespace/prefix settings. In such a situation it is usually possible to just concatenate the various output files produced by pure-gen to consolidate them into a single wrapper module. To make this easier, pure-gen provides the -N a.k.a. --noclobber option which appends the output to existing files instead of overwriting them. See the example below.
For the sake of a substantial, real-world example, here is how you can wrap the entire GNU Scientific Library in a single Pure module mygsl.pure, with the accompanying C module in mygsl.c:
rm -f mygsl.pure mygsl.c
DEFS=-DGSL_DISABLE_DEPRECATED
for x in /usr/include/gsl/gsl_*.h; do
pure-gen $DEFS -N -fc-auto -s "$x::" $x -o mygsl.pure -c mygsl.c
done
The C module can then be compiled with:
gcc $DEFS -shared -o mygsl.so mygsl.c -lgsl
Note that the GSL_DISABLE_DEPRECATED symbol must be defined here to avoid some botches with constants being defined in incompatible ways in different GSL headers. Also, some GSL versions have broken headers lacking some system includes which causes hiccups in pure-gen’s C parser. Fixing those errors or working around them through some appropriate cpp options should be a piece of cake, though.
Scott E. Dillard (University of California at Davis), Albert Graef (Johannes Gutenberg University at Mainz, Germany).