Exporting Chapel as a LibraryΒΆ

Note

This page discusses the implementation of libraries, which are still under development. The idea is to allow components to be developed in Chapel and linked with other programs and languages. It is currently fragile and difficult to work with.

Normally, Chapel assumes that you are building a main program and produces a main routine whether one is explicitly coded or not. When the --library flag is provided on the command line, it tells the Chapel compiler to produce a library instead. The user can further specify (through the --static and --dynamic flags) which type of library to produce. If neither --static nor --dynamic is specified, a platform-dependent default library type is produced.

Some platforms support linking against both static and dynamic versions of the same library. On those platforms, the --static or --dynamic flag can be used to select which type of library (and thus which kind of linking) is performed by default. Library files which are named explicitly on the chpl command line take precedence over any found through object library paths (-L). Where there is a conflict, the last library specified takes precedence.

When creating a library file from Chapel code, only those symbols with export attached to them will be available from outside the library. For example, if one defines a Chapel file foo.chpl like this:

// This function will be available to the library user
export proc bar(): int {
  // Does something
  ...
}

// As will this one
export proc baz(int x) {
  // Does something different
  ...
}

// but this function will not be
proc gloop() {
  // Does something else
  ...
}

When using a Chapel library file in C code, a fairly exact incantation is required. First, a header file must be created for the library. This does not happen through use of the Chapel --library flag during compilation! If during compilation of the library the --savec <dir> flag is also used, the C definitions for the exported functions should be visible in a C file under dir of the same name as the library Chapel file. A declaration version of these definitions should go into a .h file. For instance, if one compiled foo.chpl like this:

chpl --library --savec cDir --static -o libfoo foo.chpl

then under cDir/foo.c one might see something resembling:

int64_t bar(void) {
  ...
}

void baz(int64_t x) {
  ...
}

from which one would create a .h file:

int64_t bar(void);
void baz(int64_t x);

It may be necessary to #include some Chapel headers, such as chpl-string.h, depending on the types used in your Chapel library.

Next, if compiling dynamically, update the $LD_LIBRARY_PATH environment variable to include the directory where the new library file lives and the directory where the Chapel build lives. The latter can be found by looking at the output of a $CHPL_HOME/util/printchplenv call and finding the appropriate directory under $CHPL_HOME/lib; the directory name can be found by running $CHPL_HOME/util/printchplenv --runtime.

# Replace $PWD with the appropriate path to your library file if it isn't
# in the current directory
# Replace libDir with the directory we just found.
export LD_LIBRARY_PATH=$PWD:$CHPL_HOME/lib/`$CHPL_HOME/util/printchplenv --runtime`:$LD_LIBRARY_PATH

When using a Chapel library from C, one must first initialize the Chapel runtime and standard modules. This is done by the addition of a couple of extern declarations, calling the function chpl_library_init() before the Chapel library function calls and by calling chpl_library_finalize() after all the Chapel library function calls are finished. For example:

#include "foo.h"

extern void chpl_library_init(int argc, char* argv[]);
extern void chpl_library_finalize(void);

int main(int argc, char* argv[]) {
    chpl_library_init(argc, argv);

    baz(7); // Call into a library function

    chpl_library_finalize();

    return 0;
}

Finally, compilation of the C program involves some additional command line includes and links. The easiest way to get the right combination is by using the compileline --compile and compileline --libraries tools we provide. The compilation command would then look like this (replacing myprog.c with the name of your C program):

`$CHPL_HOME/util/config/compileline --compile` myprog.c -L. -lfoo `$CHPL_HOME/util/config/compileline --libraries`

Chapel library files cannot be used from Chapel code. The library files must include the chapel runtime and standard modules for use in a non-Chapel program and when the library is linked to a Chapel program this leads to multiple definitions of these functions.

As mentioned above, this feature is not very sturdy. Please refer to Reporting Chapel Issues if any problems are encountered.