title | author | tags |
---|---|---|
Build Systems and Packaging |
Peter Hill |
build-systems C C++ Fortran Python packaging practical |
How do you build your code? If your project is a single file, you might just run the compiler directly (or search your terminal history for the last time you ran it). Once the project grows to a handful of files, you'll want to find a better way of building everything, which is where build systems step in!
You can get the slides for the talk here
- For Python, pretty easy: setuptools and pip
- For compiled languages, often trickier
- Compiling one file from the command line is easy
- Likewise two files, maybe three
- Tens or hundreds of files unmanageable
- So we have build scripts
- Or fancier: Makefiles
Two kinds of build systems:
- Those that everyone complains about
- Those that no one has heard of
-
What do you do when you need to use different compilers?
ifeq ($(COMPILER),gcc) # gcc flags endif ifeq ($(COMPILER),intel) # intel flags endif
-
Or different systems have the libraries you need in different places?
ifeq ($(SYSTEM),mymachine) # york flags endif ifeq ($(SYSTEM),viking) # archer flags endif
- Make is suddenly not the right tool for the job
- What we need is a build system or build system generator
- Confusingly, people use both terms to refer to different things
- Let's not get bogged down in terminology!
- Something that's take what we want to build and work out how to do it
-
You've seen it before:
$ ./configure $ make $ make install
-
If you like shell scripts, you'll love Autotools
-
Actually a family of related tools: autoheader, autoconf, automake, etc.
-
Designed to generate Makefiles portable across POSIX systems
- Not so useful if you want to also compile on Windows or other weird OSes
-
Takes care of a whole bunch of standard things:
- Different compilers, MPI, etc.
- Install locations
make clean
,make install
,make uninstall
-
Surprisingly easy to get started!
- Write a simple "Hello World" program in your favourite compiled
language
#include <iostream> int main() { std::cout << "Hello, World!\n"; }
- Try running
make hello
(assuming your file is calledhello.??
and is in C or C++) - Make
Makefile.am
with:bin_PROGRAMS = hello hello_SOURCES = hello.cpp
- Run
autoscan
. This will automagically create a file calledconfigure.scan
-- renameconfigure.scan
toconfigure.ac
- Open
configure.ac
and put:on the line afterAM_INIT_AUTOMAKE([-Wall -Werror foreign])
AC_INIT
- For Fortran, you'll also need to add
AC_PROG_FC
on the line afterAM_INIT_AUTOMAKE
as well
- For Fortran, you'll also need to add
- Run
autoreconf -fvi
- Now run
./configure
thenmake
- Out of source builds are automatically supported:
make distclean
mkdir build && cd build
../configure --prefix=$(pwd)/install
make install
install/bin/hello
- This file tells
automake
what you want to build, and what is needed to build it bin_PROGRAMS
: A list ofPROGRAMS
to install inbin
hello_SOURCES
: The list ofSOURCES
needed to buildhello
- You can add other "normal makefile" stuff here too
- e.g. Fortran dependency generation
automake
takes care of all the "usual" targets
- Makes a bare-bones
configure.scan
based on your project layout
- These
AC_*
variables are macros - They get replaced by some literal text, possibly after doing something with their arguments
- Important ones:
AC_PROG_CC
/AC_PROG_CXX
/AC_PROG_FC
: Look for a C/C++/Fortran compiler and check it works!AC_CONFIG_FILES([Makefile])
: Create a file calledMakefile
from a fileMakefile.in
- Other macros for searching for libraries and checking they work
- Find MPI, check your compiler supports C++11, F2008, etc.
autoreconf
looks for all the important input files and runs all the important autotools programs on them in the correct order- Creates
Makefile.in
fromMakefile.am
- Creates
configure
fromconfigure.ac
- Neither of these generated files are supposed to be human-readable!
- Also brings in a whole bunch of other files that we won't get into
- Takes
Makefile.in
and makesMakefile
for your actual system, your compilers, libraries, etc., along with where you want to build and install it
- The thing we actually want!
- Now we can finally compile the program
- CMake is newer than Autotools, but has been around since 2000
- CMake 3.0 introduced some nicer features in 2014
- "Modern" CMake
- CMake is a "build system generator"
- Can make Makefiles as well as a whole bunch of other types, e.g. Ninja
- Works very well with a huge range of IDEs
- Works well with dependencies, especially if they also use CMake
- Copy your simple hello world program to a new directory
- We need a
CMakeLists.txt
file with three lines:cmake_minimum_required(VERSION 3.10) project(my_hello VERSION 0.1 LANGUAGES CXX) # Or C or Fortran add_executable(hello hello.cpp)
- Now make a
build
directory andcd
into it- Prefer out-of-source builds
cmake ..
instead ofconfigure
make && ./hello
as usual- Alternatively,
cmake --build . && ./hello
- This is the equivalent of autotool's
configure
script, only in the CMake language cmake_minimum_required
sets the minimum version of CMake. You should try to use the most recent version if you canpip3 install --user cmake
!
project
defines a project and the languages it uses. CMake will find the compilers.add_executable
defines an executable and its source files
- Move the "Hello, World" bit of your program into a new function in a separate file
- Organise your project a bit like this:
+-- CMakeLists.txt +-- include | +-- libsay | +-- say.hpp +-- src +-- hello.cpp +-- libsay +-- say.cpp
hello.cpp
andsay.cpp
should both#include "libsay/say.hpp"
- Fortran doesn't need the
include
directory
We need to add a few lines to our CMakeLists.txt
:
add_library(say src/libsay/say.cpp include/libsay/say.hpp)
target_include_directories(say PUBLIC include)
add_executable(hello src/hello.cpp)
target_link_libraries(hello PRIVATE say)
CMake needs to be told what to install and where
set_property(TARGET say
PROPERTY PUBLIC_HEADER include/libsay/say.hpp)
install(TARGETS hello say
EXPORT libsay
ARCHIVE DESTINATION lib
LIBRARY DESTINATION lib
RUNTIME DESTINATION bin
PUBLIC_HEADER DESTINATION include)
- CMake takes options with
-D
, such as-DCMAKE_INSTALL_PREFIX=$(pwd)/install
(to install files under./install
) or-DCMAKE_BUILD_TYPE=Debug
(for debug flags) - List all options with
cmake -LH
- Try just running
make install
again from your build directory! ccmake
is a slightly fancier TUI
- Creates a new library as a target. We can control whether its
built as a shared or static library either with the explicit
SHARED
/STATIC
keywords or withBUILD_SHARED_LIBS
option
- Sets the "include directories" property of its target, and whether
we only need them to build the target itself (
PRIVATE
) or if we also need them when we want to use the target (PUBLIC
)
- Tells CMake to link the target against the listed libraries. This can be another CMake target or an external library
- This adds all the information about the library to the target, e.g. the include directories
- Sets further properties on a target or other object
- Just lists what targets should be installed and where to
ARCHIVE
for static librariesLIBRARY
for shared librariesRUNTIME
for binariesPUBLIC_HEADER
for headers
- New comer, first release 2013
- Python-like syntax
- Very fast, simple things are very simple
- Uses Ninja build system rather than Makefiles
- Can automatically fetch and compile dependencies through its "wrap" system
- Start off with your simple "Hello world" single file
- Make
meson.build
with the following two lines:project('hello', 'cpp') # or 'c' or 'fortran' executable('hello', 'hello.cpp')
- Create a build directory:
$ meson build
- From the build directory, run ninja:
$ ninja && ./hello
- Copy your librarified "hello world"
- Update your
meson.build
fileincdir = include_directories('include') lib = shared_library('say', 'src/say/say.cpp', include_directories: incdir) executable('hello', 'hello.cpp', link_with: lib, include_directories: incdir)
- Defines a project and what languages it uses
- Defines an executable and its source files
- Targets in Meson are immutable: you have to define all their
properties when you create them
- What libraries to link against
- What directories to include
- Defines directories to be included
- Defines a shared library
- Python packaging (mostly) a lot simpler
- Has it's own complications
- Write a
setup.py
at the top-level - Enables installing with
pip
+-- setup.py
+-- hello
+-- __init__.py
+-- hello.py
# __init__.py
from .hello import hello
# hello.py
def hello():
print("Hello, World!")
from setuptools import setup
setup(name="hello",
version="0.1",
packages=["hello"],
)
- Now you can install with
pip install --user -e .
-e
argument makes it "editable": no need to reinstall while you develop
setup(name="hello",
version="0.1",
packages=["hello"],
entry_points={
"console_scripts": [
"hello = hello.hello:hello"]},
)
install_requires
: other packages and their versionsextras_require
: optional packagespython_requires
: which versions of Python are required
author
description
,long_description
url
classifiers
pip3 install --user --upgrade setuptools wheel twine
python3 setup.py sdist bdst_wheel
- Makes "wheel" and tarball for distribution
twine upload dist/my-package-0.1.0*
- Uploads package to PyPI
- You'll need account first!