In this section we will introduce topics on compiling Lisp programs. ECL is especially powerful on combining lisp programs with C programs. You can embed ECL as a lisp engine in C programs, or call C functions via Foreign Function Interface. We explain file types generated by some compilation approaches. For the examples, a GNU/Linux system and gcc as a development environment are assumed.
You can generate the following files with ECL:
Relations among them are depicted below:
Figure 3.1: Build file types
ECL provides two compilers (bytecodes compiler, and C/C++ compiler). Portable FASL files are built from source lisp files by the bytecodes compiler. Generally FASC files are portable across architectures and operating systems providing a convenient way of shipping portable modules. Portable FASL files may be concatenated, what leads to bundles. FASC files are faster to compile, but generally slower to run.
;; install bytecodes compiler
(ext:install-bytecodes-compiler)
;; compile hello.lisp file to hello.fasc
(compile-file "hello1.lisp")
(compile-file "hello2.lisp")
;; reinitialize C/C++ compiler back
(ext:install-c-compiler)
;; FASC file may be loaded dynamically from lisp program
(load "hello1.fasc")
;; ... concatenated into a bundle with other FASC
(with-open-file (output "hello.fasc"
:direction :output
:if-exists :supersede)
(ext:run-program
"cat" '("hello1.fasc" "hello2.fasc") :output output))
;; ... and loaded dynamically from lisp program
(load "hello.fasc")
If you want to make a library which is loaded dynamically from a lisp program, you should choose the fasl file format. Under the hood native fasls are just shared library files.
This means you can load fasl files with dlopen and initialize it
by calling a init function from C programs, but this is not an intended
usage. The recommended usage is to load fasl files by calling the load lisp
function. To work with Native FASL files ECL has to be compiled
with --enable-shared configure option (enabled by default).
Creating a fasl file from one lisp file is very easy.
(compile-file "hello.lisp")
To create a fasl file from more lisp files, firstly you have to compile
each lisp file into an object file, and then combine them with
c:build-fasl.
;; generates hello.o
(compile-file "hello.lisp" :system-p t)
;; generates goodbye.o
(compile-file "goodbye.lisp" :system-p t)
;; generates hello-goodbye.fas
(c:build-fasl "hello-goodbye"
:lisp-files '("hello.o" "goodbye.o"))
;; fasls may be built from mix of objects and libraries (both shared and
;; static)
(c:build-fasl "mixed-bundle"
:lisp-files '("hello1.o" "hello2.a" "hello3.so"))
Object files work as an intermediate file format. If you want to compile
more than two lisp files, you might better to compile with a
:system-p t option, which generates object files (instead of a
fasl).
On linux systems, ECL invokes gcc -c to generate object files.
An object file consists of some functions in C:
Consider the example below.
(defun say-hello () (print "Hello, world"))
During compilation, this simple lisp program is translated into the C program, and then compiled into the object file. The C program contains two functions:
static cl_object L1say_hello:
’say-hello’ function
ECL_DLLEXPORT void _eclwm2nNauJEfEnD_CLSxi0z(cl_object flag):
initialization function
In order to use these object files from your C program, you have to call
initialization functions before using lisp functions (such as
say-hello). However the name of an init function is seemed to be
randomized and not user-friendly. This is because object files are not
intended to be used directly.
ECL provides other user-friendly ways to generate compiled lisp programs (as static/shared libraries or executables), and in each approach, object files act as intermediate files.
ECL can compile lisp programs to static libraries, which can be linked
with C programs. A static library is created by
c:build-static-library with some compiled object files.
;; generates hello.o
(compile-file "hello.lsp" :system-p t)
;; generates goodbye.o
(compile-file "goodbye.lsp" :system-p t)
;; generates libhello-goodbye.a
(c:build-static-library "hello-goodbye"
:lisp-files '("hello.o" "goodbye.o")
:init-name "init_hello_goodbye")
When you use a static/shared library, you have to call its init function. The
name of this function is specified by the :init-name option. In this
example, it is then init_hello_goodbye. The usage of this function is
shown below:
#include <ecl/ecl.h>
extern void init_hello_goodbye(cl_object cblock);
int
main(int argc, char **argv)
{
/* setup the lisp runtime */
cl_boot(argc, argv);
/* call the init function via ecl_init_module */
ecl_init_module(NULL, init_hello_goodbye);
/* ... */
/* shutdown the lisp runtime */
cl_shutdown();
return 0;
}
Because the program itself does not know the type of the init function,
a prototype declaration is inserted. After booting up the lisp
environment, it invokes init_hello_goodbye via
ecl_init_module. init_hello_goodbye takes an argument,
and ecl_init_module supplies an appropriate one. Now that the
initialization is finished, we can use functions and other stuff defined
in the library.
DEPRECATED read_VV - equivalent to ecl_init_module
ECL supports the generation of executable files. To create a standalone
executable from a lisp program, compile all lisp files to object
files. After that, calling c:build-program creates the
executable:
;; generates hello.o
(compile-file "hello.lsp" :system-p t)
;; generates goodbye.o
(compile-file "goodbye.lsp" :system-p t)
;; generates hello-goodbye
(c:build-program "hello-goodbye"
:lisp-files '("hello.o" "goodbye.o"))
Like with native FASL, the program may be built also from libraries.
In this section, some file types that can be compiled with ECL were introduced. Each file type has an adequate purpose:
load lisp function
ECL provides a high-level interface c:build-* for each native
format. In case of Portable FASL the bytecodes compiler is needed.
For larger systems involving more complex file dependencies, or for systems that are portable across different Common Lisp implementations, it may be better to define systems using asdf.
ECL provides a useful extension for asdf called asdf:make-build, which
offers an abstraction for building libraries directly from system definitions.
Note that this extension is only available in the ASDF that is shipped with
ECL; it may not be available from an ASDF installed from the system or from
Quicklisp.
To download dependencies you may use
Quicklisp to load your system (with
dependencies defined). Make sure you can successfully load and run your
library in the ECL REPL (or *slime-repl*). Don’t worry about
other libraries loaded in your image – ECL will only build and pack
libraries your project depends on (that is, all dependencies you put in
your .asd file, and their dependencies - nothing more, despite
the fact that other libraries may be loaded).
An example project is included in the ECL source distribution in the
examples/asdf_with_dependence/ directory.
This project depends on the alexandria library and consists of a system
definition (example-with-dep.asd), package definition
(package.lisp), and the actual library code (example.lisp).
Before following the steps below, you must
configure ASDF to find your systems.
You can either copy or symlink the example directory in one of the standard
ASDF locations, or push the path of the example directory to your
asdf:*central-registry*, for example:
(push "./" asdf:*central-registry*)
Use this in the REPL to make an executable:
(asdf:make-build :example-with-dep
:type :program
:move-here #P"./"
:epilogue-code '(progn (example:test-function 5)
(si:exit)))
Here the :epilogue-code is executed after loading our library;
we can use arbitrary Lisp forms here. You can also put this code in
your Lisp files and directly build them without this
:epilogue-code option to achieve the same result. Running the
program in a console will display the following and exit:
Factorial of 5 is: 120
To build a static library, use:
(asdf:make-build :example-with-dep
:type :static-library
:move-here #P"./"
:monolithic t
:init-name "init_example")
This will generate example-with-dep--all-systems.a in the
current directory which we need to initialize with the
init_example function. Compile it using:
gcc test.c example-with-dep--all-systems.a -o test-static -lecl
Then run it:
./test-static
This will show:
Factorial of 5 is: 120
Note we don’t need to pass the current path in LD_LIBRARY_PATH
here, since our Lisp library is statically bundled with the executable.
The result is the same as the shared library example above. You can also
build all dependent libraries separately as static libraries.
ECL supports cross compiling Lisp files for a target system that differs from the host system on which the compilation takes place. This section of the manual describes how to use this feature and explains important things to keep in mind when cross compiling Common Lisp.
To get started, follow the steps described below:
c:read-target-info on this file.
:target option of
compile-file or with-compilation-unit.
For an example, consider two files a.lisp and b.lisp which are supposed to be linked into a shared library. This can be accomplished with the following steps:
(defvar *target* (c:read-target-info "/path/to/ecl/installation/lib/ecl-xx.x.x/target-info.lsp"))
(compile-file "a.lisp" :target *target* :system-p t)
(load "a.lisp") ; make macro definitions in "a.lisp" accessible to "b.lisp"
(compile-file "b.lisp" :target *target* :system-p t)
(with-compilation-unit (:target *target*)
(c:build-shared-library "example"
:lisp-files '("a.o" "b.o")
:init-name "init_example"))
Cross compilation using ASDF is not supported yet but is planned for the future.
The ubiquity with which typical Common Lisp programs run code at compile
time in macros or other similar constructs makes cross compilation
somewhat more challenging than in other programming languages. Since
compilation happens on a different system than the one in which the code
generated by a macro is run, differences between the host and target
environment can lead to bugs if the macro hasn’t been written with cross
compilation in mind. Consider for instance the following example macro
iterating over prime number in the range from a to b:
(defmacro do-prime-numbers ((a b) p &body body)
`(loop with ,p = (next-prime ,a)
until (> ,p ,b)
do ,(if (typep b 'fixnum)
`(locally (declare (fixnum ,p))
,@body)
`(progn ,@body))
(setf ,p (next-prime (1+ ,p)))))
While this macro works fine in a standard setting, it will produce
an incorrect declaration when cross compiling if the size of a fixnum
is larger in the host than in the target and the upper bound b is
larger than most-positive-fixnum in the target but smaller than
most-positive-fixnum in the host. In general, any observable
difference between target and host system may contribute to such issues.
In the list below, we collect a number of such potential issues and
explain strategies to solve them.
(defmacro do-prime-numbers ((a b) p &body body)
`(loop with ,p = (next-prime ,a)
until (> ,p ,b)
do (locally (declare (type (integer ,a ,b) ,p))
,@body)
(setf ,p (next-prime (1+ ,p)))))
If the declared integer type is a subtype of fixnum, the ECL compiler will automatically take this into account and optimize the same as if a fixnum declaration had been made.
(defmacro do-prime-numbers ((a b) p &body body &environment env)
`(loop with ,p = (next-prime ,a)
until (> ,p ,b)
do ,(if (typep b 'fixnum env)
`(locally (declare (fixnum ,p))
,@body)
`(progn ,@body))
(setf ,p (next-prime (1+ ,p)))))
It is important, however, not to blindly modify every occurrence of
typep and subtypep in this way. Some calls to these
functions may check for types in the host environment, in which case
they should not receive an environment parameter. It is necessary to
know for which environment the call is meant in order to decide what to
do.
*features*, leading to mismatched #+
and #- read time conditionals. During cross compilation,
*features* will be rebound to the value in the target
system.4 However, macros that are defined in the host system
will see the host *features* during read time. Consider again the
example given above where a.lisp was cross compiled, then loaded
before b.lisp was cross compiled. If a.lisp contains a
macro definition
(defmacro my-macro (...) #+android `(do-something ...) #-android `(do-something-else ...))
then my-macro will expand to do-something in a.lisp
but do-something-else in b.lisp if cross compiling for the
android target. There are several ways to deal with this issue:
*features* has the same value as in the target system. This can
be accomplished either by wrapping load in a
with-compilation-unit call with a :target option given
or by using the :load keyword option of compile-file in
conjunction with :target. In the above example, we would compile
as follows:
(compile-file "a.lisp" :target *target* :system-p t :load t) (compile-file "b.lisp" :target *target* :system-p t)
Note that this will not work if the file contains read time conditionals selecting between different compile time code paths due to differences in the host system (let’s say for example some code generation from a file that is located in a different location for different compilation hosts).
(defmacro my-macro (...)
(if (member :android *features*)
`(do-something ...)
`(do-something-else ...)))
*features* in the
host before compilation. This is best suited for conditionals which only
add but not subtract code (keywords which only appear in #+
conditionals). In our example, we would call
(push :android *features*)
before starting cross compilation.
Another option to avoid the aformentioned issues entirely is to use emulation instead of cross compilation. Even if full-blown emulation is too complicated, simply cross compiling from a version of ECL compiled for the same word size as the target system would for instance avoid the fixnum issues in our example macro alltogether.
Whether the compiler is switched to cross compilation mode or not can be
diagnosed from the presence of a :cross keyword in
*features*. If necessary, this can be used to select different
code paths as well.
ECL provides some global variables to customize which C compiler and compiler options to use:
It is not required to surround the compiler flags with quotes or use slashes before special characters.
string c:*user-cc-flags* ¶Flags and options to be passed to the C compiler when building FASL, shared libraries and standalone programs.
string c:*user-linker-flags* ¶Flags for options (e.g. -Wl,foo flags, usually in the
$LDFLAGS variable in autoconf) to be passed to the linker when
building FASL, shared libraries and standalone programs.
string c:*user-linker-libs* ¶Flags for libraries (e.g. -lfoo flags, usually in the
$LIBS variable in autoconf) to be passed to the linker when
building FASL, shared libraries and standalone programs.
string c:*cc-optimize* ¶Optimize options to be passed to the C compiler.
string c:*user-ld-flags* ¶DEPRECATED Flags and options to be passed to the linker when building FASL, shared libraries and standalone programs.
string c::*cc* ¶This variable controls how the C compiler is invoked by ECL. One can set the variable appropriately adding for instance flags which the C compiler may need to exploit special hardware features (e.g. a floating point coprocessor).
string c::*ld* ¶This variable controls the linker which is used by ECL.
string c::*ranlib* ¶Name of the ‘ranlib’ program on the hosting platform.
string c::*ar* ¶Name of the ‘ar’ program on the hosting platform.
string c::*ecl-include-directory* ¶Directory where the ECL header files for the target platform are located.
string c::*ecl-library-directory* ¶Directory where the ECL library files for the target platform are located.
You may also link ECL runtime statically. That is not covered in this walkthrough.
Changes to *features* during compilation are
carried over in the target information structure. This means that in a
scenario where one cross compiles a file a.lisp containing
(eval-when (:compile-toplevel) (push :my-feature *features*))
before cross compiling another file b.lisp containing
#+my-feature, the read time conditional in b.lisp will
evaluate to true.