Makefile¶
The most primitive build system a package can use is a plain Makefile. Makefiles are simple to write for small projects, but they usually require you to edit the Makefile to set platform and compiler-specific variables.
Phases¶
The MakefileBuilder and MakefilePackage base classes come with 3 phases:
edit- edit the Makefilebuild- build the projectinstall- install the project
By default, edit does nothing, but you can override it to replace hardcoded Makefile variables.
The build and install phases run:
$ make
$ make install
Important files¶
The main file that matters for a MakefilePackage is the Makefile.
This file will be named one of the following ways:
GNUmakefile (only works with GNU Make)
Makefile (most common)
makefile
Some Makefiles also include other configuration files.
Check for an include directive in the Makefile.
Build system dependencies¶
Spack assumes that the operating system will have a valid make utility installed already, so you don’t need to add a dependency on make.
However, if the package uses a GNUmakefile or the developers recommend using GNU Make, you should add a dependency on gmake:
depends_on("gmake", type="build")
Types of Makefile packages¶
Most of the work involved in packaging software that uses Makefiles involves overriding or replacing hard-coded variables. Many packages make the mistake of hard-coding compilers, usually for GCC or Intel. This is fine if you happen to be using that particular compiler, but Spack is designed to work with any compiler, and you need to ensure that this is the case.
Depending on how the Makefile is designed, there are 4 common strategies that can be used to set or override the appropriate variables:
Environment variables¶
Make has multiple types of assignment operators.
Some Makefiles use = to assign variables.
The only way to override these variables is to edit the Makefile or override them on the command-line.
However, Makefiles that use ?= for assignment honor environment variables.
Since Spack already sets CC, CXX, F77, and FC, you won’t need to worry about setting these variables.
If there are any other variables you need to set, you can do this in the setup_build_environment method:
def setup_build_environment(self, env: EnvironmentModifications) -> None:
env.set("PREFIX", prefix)
env.set("BLASLIB", spec["blas"].libs.ld_flags)
cbench is a good example of a simple package that does this, while esmf is a good example of a more complex package.
Command-line arguments¶
If the Makefile ignores environment variables, the next thing to try is command-line arguments.
You can do this by overriding the build_targets attribute.
If you don’t need access to the spec, you can do this like so:
build_targets = ["CC=cc"]
If you do need access to the spec, you can create a property like so:
@property
def build_targets(self):
spec = self.spec
return [
"CC=cc",
f"BLASLIB={spec['blas'].libs.ld_flags}",
]
cloverleaf is a good example of a package that uses this strategy.
Edit Makefile¶
Some Makefiles are just plain stubborn and will ignore command-line variables.
The only way to ensure that these packages build correctly is to directly edit the Makefile.
Spack provides a FileFilter class and a filter method to help with this.
For example:
def edit(self, spec, prefix):
makefile = FileFilter("Makefile")
makefile.filter(r"^\s*CC\s*=.*", f"CC = {spack_cc}")
makefile.filter(r"^\s*CXX\s*=.*", f"CXX = {spack_cxx}")
makefile.filter(r"^\s*F77\s*=.*", f"F77 = {spack_f77}")
makefile.filter(r"^\s*FC\s*=.*", f"FC = {spack_fc}")
stream is a good example of a package that involves editing a Makefile to set the appropriate variables.
Config file¶
More complex packages often involve Makefiles that include a configuration file. These configuration files are primarily composed of variables relating to the compiler, platform, and the location of dependencies or names of libraries. Since these config files are dependent on the compiler and platform, you will often see entire directories of examples for common compilers and architectures. Use these examples to help determine what possible values to use.
If the config file is long and only contains one or two variables that need to be modified, you can use the technique above to edit the config file. However, if you end up needing to modify most of the variables, it may be easier to write a new file from scratch.
If each variable is independent of each other, a dictionary works well for storing variables:
def edit(self, spec, prefix):
config = {
"CC": "cc",
"MAKE": "make",
}
if spec.satisfies("+blas"):
config["BLAS_LIBS"] = spec["blas"].libs.joined()
with open("make.inc", "w") as inc:
for key in config:
inc.write(f"{key} = {config[key]}\n")
elk is a good example of a package that uses a dictionary to store configuration variables.
If the order of variables is important, it may be easier to store them in a list:
def edit(self, spec, prefix):
config = [
f"INSTALL_DIR = {prefix}",
"INCLUDE_DIR = $(INSTALL_DIR)/include",
"LIBRARY_DIR = $(INSTALL_DIR)/lib",
]
with open("make.inc", "w") as inc:
for var in config:
inc.write(f"{var}\n")
hpl is a good example of a package that uses a list to store configuration variables.
Variables to watch out for¶
The following is a list of common variables to watch out for. The first two sections are implicit variables defined by Make and will always use the same name, while the rest are user-defined variables and may vary from package to package.
Compilers
This includes variables such as
CC,CXX,F77,F90, andFC, as well as variables related to MPI compiler wrappers, likeMPICCand friends.Compiler flags
This includes variables for compiler flags, such as
CFLAGS,CXXFLAGS,F77FLAGS,F90FLAGS,FCFLAGS, andCPPFLAGS. These variables are often hard-coded to contain flags specific to a certain compiler. If these flags don’t work for every compiler, you may want to consider filtering them.Variables that enable or disable features
This includes variables like
MPI,OPENMP,PIC, andDEBUG. These flags often require you to create a variant so that you can either build with or without MPI support, for example. These flags are often compiler-dependent. You should replace them with the appropriate compiler flags, such asself.compiler.openmp_flagorself.compiler.pic_flag.Platform flags
These flags control the type of architecture that the executable is compiled for. Watch out for variables like
PLATorARCH.Dependencies
Look out for variables that sound like they could be used to locate dependencies, such as
JAVA_HOME,JPEG_ROOT, orZLIBDIR. Also watch out for variables that control linking, such asLIBS,LDFLAGS, andINCLUDES. These variables need to be set to the installation prefix of a dependency, or to the correct linker flags to link to that dependency.Installation prefix
If your Makefile has an
installtarget, it needs some way of knowing where to install. By default, many packages install to/usror/usr/local. Since many Spack users won’t have sudo privileges, it is imperative that each package is installed to the proper prefix. Look for variables likePREFIXorINSTALL.
Makefiles in a sub-directory¶
Not every package places their Makefile in the root of the package tarball.
If the Makefile is in a sub-directory like src, you can tell Spack where to locate it like so:
build_directory = "src"
Manual installation¶
Not every Makefile includes an install target.
If this is the case, you can override the default install method to manually install the package:
def install(self, spec, prefix):
mkdir(prefix.bin)
install("foo", prefix.bin)
install_tree("lib", prefix.lib)
External documentation¶
For more information on reading and writing Makefiles, see: https://www.gnu.org/software/make/manual/make.html