Python and C interface¶
How is the C interface exported¶
Featomic exports a C interface, created directly in Rust without involving any C code.
This is done by marking functions as #[no_mangle] extern pub fn <XXX>
in
featomic/src/c-api/*.rs
, and only using types safe to send to C (mostly
pointers and basic values such as floats or integers). Of these markers, pub
ensures that the function is exported from the library (it should appear as a
T
symbol in nm
output); extern
forces the function to use the C
calling convention (a calling convention describes where in memory/CPU registers
the caller should put data that the function expects); and #[no_mangle]
tells the compiler to export the function under this exact name, instead of
using a mangled named containing the module path and functions parameters.
Additionally, the C interfaces expose C-compatible structs declared with
#[repr(C)] pub struct <XXX> {}
; where #[repr(C)]
ensures that the
compiler lays out the fields in the exact order they are declared, without
re-organizing them.
featomic
is then compiled to a shared library (libfeatomic.so
/
libfeatomic.dylib
/ libfeatomic.dll
), which can be used by any language
able to call C code to call the exported functions without ever realizing it is
speaking with Rust code.
The list of exported functions, together with the types of the function’s
parameters, and struct definitions are extracted from the rust source code using
cbindgen, which creates the featomic/include/featomic.h
header file
containing all of this information in a C compatible syntax. All of the
documentation is also reproduced using doxygen syntax.
How does the Python interface works¶
The Python interface used the ctypes module to call exported symbols from the
shared library. For the Python code to be able to call exported function safely,
it needs to know a few things. In particular, it needs to know the name of the
function, the number and types of parameters and the return type of the
function. All this information is available in featomic/include/featomic.h
,
but not in a way that is easily accessible from ctypes. There is a script in
python/scripts/generate-declaration.py
which reads the header file using
pycparser, and creates the python/featomic/_c_api.py file which
declares all functions in the way expected by the ctypes module. You will
need to manually re-run this script if you modify any of the exported functions
in featomic/src/c-api.
The schematic below describes all the relationships between the components involved in creating the Python interface.