-
Notifications
You must be signed in to change notification settings - Fork 2
kugel-/kmake
Folders and files
Name | Name | Last commit message | Last commit date | |
---|---|---|---|---|
Repository files navigation
===== kmake ===== .. contents:: kmake is an Automake replacement. It is based purely on GNU make and generates rules and recipes dynamically during make time, avoiding the need to generate Makefile files. It is in many ways similar to kbuild of the Linux kernel, but it is designed to build user space programs, static and shared libraries. Goals ----- kmake is targeted to replace Automake in an autotools-based build system. It is not a drop-in replacement and it still depends on other components such as libtool. Global dependency tracking ^^^^^^^^^^^^^^^^^^^^^^^^^^ Automake uses recursive make by default, i.e. there are separate make invocations for each directory (separate processes). Dependencies cannot be tracked across those invocations, leading to incomplete builds if one wants to update only subdirectories. While it's possible to use a fully non-recursive setup with Automake, it is very troublesome. Automake doesn't have directory-level CFLAGS. Adding to AM_CFLAGS changes CFLAGS globally, so you have to fall back to per-target CFLAGS which quickly requires lots of more lines of Automake code and is hard to maintain in larger scale. Additionally, each sub-Makefile.am needs to know it's exact location in the project, unless you use %C% and %D% helpers which quickly render the Makefile.am files unreadable for humans. kmake on the other hand always uses a non-recursive setup. Sub-makefiles are included recursively, forming one big Makefile (in memory) which has the complete picture about everything that can be built. It adds directory-level CFLAGS so you don't have to give up on modularity. Components can be separated by directory and maintained individually, but of course components can depend on other components and kmake ensures those dependencies are applied properly. If really needed, kmake provides a way to invoke a separate make process for some directories, but this is meant as a fall back mostly for 3rd party components. Modularity ^^^^^^^^^^ kmake tries hard to allow projects to remain modular. Instead of one large Makefile there are many sub-makfiles files. Each can specify its own programs, libs and data files, its own CFLAGS and library dependencies. It goes further by allowing to specify source files relative to the sub-makefile. Finally, some directories may even require their own make invocation, kmake supports this. All of this makes it usually harder to maintain modularity in a non-recursive make setup. However, kmake goes great lengths to achieve this. Avoid generating Makefiles ^^^^^^^^^^^^^^^^^^^^^^^^^^ One weakness of Automake is the fact that it generates files, and the generated Makefiles contain rules to recreate themselves. When one manages to introduce a syntax problem to the Makefiles, all hope is lost. Due to the syntax problem, the Makefiles cannot be updated. Often enough it's easier to wipe the build tree. With kmake there is no such step to generate files. The sub-makefiles are included at make-time so syntax errors can be corrected immediately. There is no additional dependency since it's based on pure GNU make code. Re-compile on compiler command line changes ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ Compiler options affect the build output in many ways, so naturally a C file must be recompiled of a compiler option is added or removed. This works for all possible ways to specify the options, including the ones that you can pass on the make invocation (e.g. make CFLAGS=-g). Concise sub-makefiles ^^^^^^^^^^^^^^^^^^^^^ Makefile.am files are mostly compact, unless you need lots per target variables, then they can quickly become full of duplicate lines. Because kmake supports directory-level CFLAGS (and other flags, including custom flags that are not known to kmake initially), the sub-makefiles remain concise. To further aid compactness, kmake supports one-line ("inline") conditionals similar to kbuild. Extensibility ^^^^^^^^^^^^^ kmake can be extended at make-time by so called generators. These add rules and recipes for custom targets that kmake doesn't support out of the box. This can be used to tell kmake how to generate source files with the help of external commands. If a build target depends on such a generated file kmake introduces the proper dependency so that the file is generated as needed. Additionally, if a target doesn't explicitly lists a generated file as dependency, but instead lists its directory in the INCLUDES flag, kmake will add a dependency as well. This is particularly useful for generated headers. Those generators can benefit from the same mechanism that re-compiles C files when CFLAGS changes. A properly implemented generator re-generates output if any of the options to the external command changes. Generators can be chained, which makes it possible teach kmake to generate and build arbitrary file formats, without having to change kmake core itself. Fundamental Concepts -------------------- kmake uses terms for some concepts. Since these are used in the rest of the document they shall be explained first. Variables (vars) ^^^^^^^^^^^^^^^^ The key element in kmake are variables, or just short vars. kmake defines some but also learns new ones dynamically (see `Adding vars`_). Vars generally have a associated list via the -y suffix and may have properties (see `Properties`_). via other suffixes. The associated list usually holds files, but can also list options or other things. How the list is interpreted depends on the variable. Pre-defined variables include: `subdir` This one lists sub-directories where kmake looks for sub-makefiles, see `Recursive inclusion of subdir.mk`_ `bin`, `sbin` These list programs that must be built and installed. `libs` This one lists libraries, static or shared, libraries that must be built and installed. `data`, `sysconf` These list data files that must be installed `tests` This one lists test programs. These variable list files that must be installed (except for tests) and usually they need to be built before. In fact, these files become targets. subdir.mk add targets to the vars by assigning to their list (`var`-y): ``bin-y := program`` basically states that there is a target named program that must be built and installed. You may dynamically add more variables, for example if you need to install in a non-standard location, see `customization and extensions`_ for defining new vars and `Properties` on how to select the install directory. Targets ^^^^^^^ Arguably, this is what kmake is about: defining targets and updating them as necessary. kmake learns about targets by listing them in one of the variables. Whenever that is done, this implicitly creates target variables. The name matches that of the target, and being a variable it has a file list (`target`-y) and may have properties (see `Properties`_). There are no predefined target variables, kmake learns about them dynamically. For target variables, `target`-y usually lists source files. If no such list exists and the target variable defines (or inherits) the suffix property, then kmake attempts to find exactly one source file that matches the name of the target plus its suffix property value. If the target has a suffix on its own, it's replaced. Additionally, `target`-y can list library targets that are defined in the same subdir.mk file. Here's an example for a target specification:: bin-y := tool libs-y := libmy.a libmy.a-y := libmy.c tool-y := tool.c tool-y += libmy.a With the suffix property, this can be shortened a bit:: bin-suffix := .c bin-y := tool # tool.c automatically picked up Properties ^^^^^^^^^^ Variables (including target vars) can have properties. Properties are inherited by targets. That means a target automatically gets the property value of the var that the target is listed under. But targets can be override suffixes with specific values. Pre-defined properties include: `-compiler` The compiler program to use to build the targets, for example to use clang on a per-var basis. `-libtooltag` When libtool is used to build shared libraries, this property overrides the tag that libtool uses. By default kmake passes CXX or CC depending on the source files. Use this to enforce, for example, CC. See https://www.gnu.org/software/libtool/manual/html_node/Tags.html. `-dir` The installation directory used at `make install` time. The special value `noinst` indicates that the target isn't meant to be installed at all. `-driver` The command that is used to run test targets. By default, this property is not set and tests will execute within a shell spawned by make. But if it's set then kmake will invoke this driver program with the test as the first parameter. `-suffix` The suffix that is used to find the default source file if the target does not list any source file in `target`-y. By default, this property is not set. But if it's set then kmake assumes that the only source file is named `target` plus the suffix. If the target has a suffix it's replaced. As you might see, not all properties make sense on all targets and therefore some suffixes are simply ignored on some targets. Flags ^^^^^ Another class of variables are flag variables. These are used to list options that are passed to the compiler or other tools. Predefined flag vars include: `CPPFLAGS`, `CFLAGS`, `CXXFLAGS` and `LDFLAGS` Options listed are passed to the preprocessor, compilers and the linker, respectively. `INCLUDES` Include directories that are passed to the preprocessor. List only the directory with a trailing slash, kmake will prepend -I itself. In addition kmake will produce (order-only) dependencies on generated files located in such directories. This automatically ensures that headers are generated before any source file that includes such headers, although the mechanism isn't strictly restricted to headers. `LIBS` Library path and link options that are passed last to the linker. `DEPS` Additional dependencies that used during compilation and linking stages, depending on whether library our sources files are listed. The DEPS flag behaves mostly the same as `var`-y except that the paths are not relative to the current sub-makefile. for a distinction between `var`-y). One special property about flag variables is that they can be combined with targets, effectively building *target-specific flags*. The values are generally appended to the flag variables, allowing to override some options for individual targets:: CFLAGS-y := -O1 mytool-CFLAGS-y := -O2 mytool-LIBS-y := -lz Here, mytool will be compiled with both -O1 -O2, but the compiler generally uses the last occurence. Another special property is that for all flags there exists a subdir-XXX-y flavor. This flavor sets the initial flags for targets defined in sub-directories (see next section), and *only* for them. subdir-XXX-y does not affect the targets of the current subdir.mk (this is different to kbuild's subdir-ccflags-y). Therefore you probably want to initialize them like this:: CFLAGS-y := -O1 subdir-CFLAGS-y := $(CFLAGS-y) -Werror Recursive inclusion of subdir.mk ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ Besides variables, one more fundamental concept is the recursive inclusion of sub-makefiles, which must be named subdir.mk each. Instead of spelling out one large Makefile, kmake reads many subdir.mk files in order to dynamically generate that large Makefile only in memory. The subdir.mk files itself are concise and specify only local targets. Example ``subdir.mk``:: subdir-y := sub/ dir/ CFLAGS-y := -O2 bin-y := program1 program2 sbin-y := program3 libs-y := libmy.a program2-suffix := .cpp program3-y := othername.c libmy.a-CPPFLAGS-y := -DHAVE_FOO libmy.a-y := my.c subdir.mk generally don't add to global variables. They assign to some var, and then setup target vars. They also may set flag vars or target-specific flags as required. Crucially, they also assign sub-directories to the subdir-y list. This will cause kmake to recursively read subdir.mk files in a depth-first fashion. Please be note that the trailing slash following the directory names is mandatory. Between including two subdir.mk, kmake resets all kinds of vars (except target vars) to empty values, so no subdir.mk has the chance to accidentally delete targets defined in another subdir.mk or otherwise affect them negatively. Note that this is also true for subdir.mk files in a sub-directory. Flags, such as CFLAGS and friends, do *not* affect sub-directories in any way. To set flags for sub-directories, use subdir-CFLAGS-y. subdir-CFLAGS-y is reset itself as well for sub-directory subdir.mk files, so they may add even more compiler options by defining their own subdir-CFLAGS-y. kmake will ensure leaf directories will see the concatenation of all subdir-CFLAGS-y in the parent directiroes. Again, remember that subdir-CFLAGS-y does only affect sub-directories but not the current one. However, target vars not reset since other subdir.mk files shouldn't know about them. This may be used to have multiple subdir.mk contribute to a global target but it may be more appropriate to implement this with static libraries instead. The var values provided by all subdir.mk contribute to a number of lists. For example `all_bin` lists all programs that have been assigned to `bin`-y across all subdir.mk files. Likewise, there are all_sbin and all_libs lists and many more. This is how kmake remembers all possible targets in order to generate rules after processing all subdir.mk files. Generators ^^^^^^^^^^ So far we have covered targets that are built from C or C++ source files, as kmake primarily targets those two languages. However, since one of kmake's stated goes is extensibility, there is the concept of `generators`. That is, a kind of plugins that enable kmake to build targets that it otherwise doesn't know about. Originally, this was designed to allow for running programs which in turn generate source code files. Therefore, the output of generators can be input of other targets and even other generators. In fact, generators can be changed arbitrarily, for example to pre- and post-process files with sed. Please note that the generators are not executed unless the output file is a prerequisite of some other target or is to be installed itself. kmake ships two real generators (for calling sed and cat) which are not enabled by default. They are enabled by including the files `gen-sed.mk` and `gen-cat.mk` respectively. Two pseudo generators, `byproduct` and `generated`, are also included enabled by default. gen-sed The sed generator allows you to feed a file into sed in order to generate a processed version of it. You can use for any number of output files, and each output file can specify its own SED_SCRIPT that defines the script that is passed to sed. Here's an example:: sysconf-y := output.txt sed-y := output.txt output.txt-y := input.txt output.txt-SED_SCRIPT-y := 's,foo,bar,' This will generate out by running sed on the file input.txt to generate output.txt. Since the file output.txt is to be installed, sed will be executed if necessary: if output.txt doesn't exist, if input.txt is newer or if SED_SCRIPT is changed since the last run. gen-cat The cat generator allows you to concat a number of input files in order to generate a combined output file (using the cat program). Again, you can use the cat generator for multiple output files, each with its own set of input files. However it doesn't pass any special flag to cat. Here's an example:: bin-y := program cat-y := combined.c program-y := combined.c combined.c-y := file1.c file2.c The cat generator will be executed since a program requires the output file. The output file is generated as needed: if it doesn't exist or any of its input files is newer. byproduct The byproduct pseudo generator helps managing dependencies for programs that produce multiple output files. Because GNU make provides no universal facility to manage those (except pattern rules which require uniform names of all output files) this may be needed for race-free builds. Example:: # mygen is an imaginary program that produces foo.c and bar.h bin-y := program program-y := foo.c baz.c bar.h mygen-y := foo.c byproduct-y := bar.h bar.h-y := foo.c As you can see, by listing bar.h in byproduct-y, kmake becomes aware the file is generated as well (setting vpath if necessary among other things). Additionally, by listing foo.c in bar.h-y, kmake will emit a rule (without recipe) so that bar.h can be build explicitly as well, by generating foo.c. This is especially useful if bar.h was to be included in other C files. In this example, by listing bar.h in program-y, kmake can ensure that bar.h will exist before any object of program is build. generated The generated pseudo generator marks files as generated that are entirely generated using custom rules. By making kmake aware of such generated files it can set vpath accordingly, among other things. The custom rule will be run if any target build by kmake lists that file. Example:: bin-y := program program-y := foo.c myheader.h generated-y := myheader.h $(objdir)myheader.h: $(srcdir)myheader.h.in generate -in $< -out $@ Always use $(srcdir) and $(objdir) to prepend the proper prefix to target and prerequsite files in such custom rules, in case a `separate output directory`_ is used. You can see that generators are quite powerful. In fact, they make kmake truly extensible since generators can be dynamically added to kmake if a project has specific requirements. Details on how to create generators follow in section `customization and extensions`_. How to use kmake ---------------- Basic setup ^^^^^^^^^^^ In order to make use of kmake you must create a so-called stub-makefile that includes kmake.mk. Usually this would be your Makefile that make uses by default. The stub-makefile is also the file where you include the shipped generators, gen-sed.mk and gen-cat.mk, before including kmake.mk. Besides including kmake.mk, the Makefile may set some default parameters for kmake but it's not strictly required. The minimalistic setup looks like this, assuming you have imported kmake's source tree into a subdirectory called kmake. `Makefile`:: include kmake.mk `subdir.mk`:: bin-y := program As you can see, the most minimalistic setup requires just the stub-makefile and one subdir.mk file. Parameterization ^^^^^^^^^^^^^^^^ kmake can be parameterized through defining some global variables, either before including kmake.mk or simply by mentioning them on the make command line. Prefix `````` Prefix refers to the installation directories. In a basic setup, kmake uses the following defaults:: prefix = /usr/local/ bin-dir = /usr/local/bin sbin-dir = /usr/local/sbin libs-dir = /usr/local/lib data-dir = /usr/local/share sysconf-dir = /usr/local/etc CROSS_COMPILE = CC = cc CXX = c++ AR = ar STRIP = strip LIBTOOL = libtool INSTALL = install If you want to override the above defaults, you can define ``prefix`` to affects all directories, or define `bindir`, `sbinddir`, `libdir`, `datadir` and `sysconfdir` to override individual ones. Toolchain ````````` Toolchain selection can also be parametrized. You can define `CROSS_COMPILE` which is prefixed to the relevant toolchain programs (`CC`, `CXX`, `AR` and `STRIP`). Additionally you can also change the individual programs, e.g. to use `clang` instead of the usual system's default of `gcc`. However, be aware that `CROSS_COMPILE` is applied regardless. Tests ````` Tests are are listed in the `tests` variable. Tests are compiled and executed only on `make check`. kmake, by default, will simply execute via make recipies, meaning that they are passed to the shell. The exit status is indicated by a "PASS" or "FAIL" output. If you have lots of Python-based unit tests you may want to pass them to the python interpreter directly. To do that, set the driver property for the `tests` variables (or any custom tests variable added via `extra-tests`) or set the driver property of each python test individually. Silent / verbose output ```````````````````````` By default, kmake produces quiet and pretty output, like this:: GEN g/g.c CC a/liba.a-a.o CC a/libx.la-libx.lo To get the command lines that are executed, set ``V=1``. Some tools output even more details if you set ``V=2`` but that may produce really lots of output. Separate output directory ````````````````````````` By default, kmake places output files, like objects, library files and programs to the current directory. However, it is recommended to use a separate directory for output files, since that allows you to retain a clean source tree. Output directories are defined by setting `O=some/builddir`. Then, kmake will place every output file under that sub-directory. The directory structure of the source tree is reproduced, except if no outfile would be produced in a given directory. Partial build ````````````` kmake can be told to build only targets in specific directories. These directories are defined by setting M on the command line. For example, ``make M=foo`` builds only libraries and programs under foo/. However, this is not quite the same as building only in a sub-directory with Automake. Even with partial build, kmake will respect dependencies of each target. This means any required dependency will be updated if necessary, even if outside the selected directories. This is usually exactly what you want, as you don't need to explicitly build dependency libraries beforehand. Multiple directories are supported (separated by spaces) but remember to properly escape the space on the command line (e.g. ``M=foo\ bar``). The partial build filter also applies to check, clean and install targets. Customization and extensions ---------------------------- As has been mentioned before, kmake leaves room for extensions, because the defaults usually don't fit anyone. Adding vars ^^^^^^^^^^^ Besides overriding properties of pre-defined variables, like setting `bin-dir` to something else, you can also add vars (vars as in `Variables (vars)`_). This is helpful if you have custom installation directories but also need `bin` and `sbin` as they are. This is also useful if you build plugins for some other application and you must put them into a directory where that application looks for plugins. To add vars, define any one (or more) of the following variables, ether in any subdir.mk or in the stub-makefile: `extra-progs` Vars mentioned here list programs `extra-libs` Vars mentioned here list libraries `extra-data` Vars mentioned here list data files `extra-tests` Vars mentioned here list test programs Again, you add vars. Then you use these new vars to list targets and set properties. The following example adds a data var `src` for source code installed to /usr/src:: extra-data := src src-dir := $(prefix)src src-y := a.c b.c c.c If multiple subdir.mk files add the same var, kmake will take care in the following way: the list (`var`-y) is appended each time it is mentioned, so targets are not lost. However, the last setter of a given property takes precedence, so each subdir.mk should set the same property values. Alternatively (and recommend), you can add the var and set its properties in the stub-makefile and then only add to the var's list in the subdir.mk files (which is exactly the same way for pre-defined variables). The following ones are only relevant if you need to extend kmake with `Generators`_. `extra-gen` Vars mentioned here list generators `extra-flags`, `extra-append-flags` These variables list flag vars. They differ in the way kmake handles them vs. target-specific flags- With `extra-flags` and `extra-append-flags` you can add flag vars that behave much the same as the pre-defined ones like CFLAGS (see `Flags`). But kmake wouldn't use them in any rule so they are only useful if you use them in rules of custom generators. Adding generators ^^^^^^^^^^^^^^^^^ Generators are powerful and make kmake truly extensible. Generators can add arbitrary build rules to kmake, allowing to build just about anything. In theory they are even capable of adding support for new programming languages, though it hasn't been shown in practice. Adding generators is done primarily by listing it in `extra-gen`, in the same way as adding a var (see above). It is recommended to do this before including `kmake.mk`, so that all subdir.mk can easily use it . However, it's also perfectly OK if a generator is defined and used only in a single subdir.mk. The name you chose for the generator is significant because kmake will call out for two macros, named after the generator, when it builds up all build rules after processing all subdir.mk files. So, if the generator is named `foo`, then kmake will call the macros `foo_rule` and `foo_recipe`. The former will be called once for every target (the target is passed as the first argument), while the latter is called exactly once (the list of all targets is passed as the first argument). Optionally, you can assign the suffix property to your generator variable if you want kmake to automatically find source files for targets. Also optional is telling kmake about additional flags (see `Flags`_) that kmake shall learn via `extra-flags`. Flag vars listed here will be recorded in the same manner as CFLAGS. The flags can be used in the rule and recipe parts of the generator to target pass specific options to the program. As an example, let's look at the sed generator that ships with kmake and then detail what to do in the rule and recipe parts of a generator:: extra-gen += sed extra-flags += SED_SCRIPT define sed_rule $(OUTDIR)$(1): SED_SCRIPT = $(call getvar,$(1),SED_SCRIPT) $(OUTDIR)$(1): CMD = $$(SED_SCRIPT) $(OUTDIR)$(1): $(SRCDIR)$(call getsrc,$(1)) $(OUTDIR)$(1): $(OUTDIR)$(call getcmdfile,$(1)) endef define sed_recipe $(addprefix $(OUTDIR),$(1)): $$(call printcmd,GEN,$$@) $$(Q)sed $$(addprefix --expression=,$$(SED_SCRIPT)) $$< >$$@.tmp && ( mv $$@.tmp $$@ ; rm -f $$@.tmp ) endef sed-suffix := .c.sed Generator rule `````````````` The rule part is more kmake-specific that the recipe part, and here kmake offers some macros that it uses internally as well. $(OUTDIR) Evaluates to the output directory if it's set via ``make O=path`` Always prefix targets with $(OUTDIR), kmake doesn't do this before passing them to your macros, because some other macros require the non-prefixed version, in particular getsrc below. $(SRCDIR) Evaluates to the source directory if it's set via ``make S=path`` and make's working directory is outside of it. Always prefix source files with $(SRCDIR), getsrc (see below) doesn't do this because it would hurt other users of that macro. getsrc Basically, this gets the list of files formed by `var`-y, `var`-DEPS-y and directory-level DEPS (in that order). The value of `var`-y is subject to the automatic source file detection if the generator has a suffix property. Pass the target name ($1) without $(OUTDIR) prefix. Always add the source file list first to the prerequisites of the target. getcmdfile This is optional but it allows kmake to generate .cmd file (see `Re-compile on compiler command line changes`_) so that the output file will be regenerated when relevant flags change. kmake will write the contents of the variable $(CMD) into the .cmd file, so usually that variable is a target-specific one (in GNU make terms). getvar This macro retrieves the target-specific value of a flag/var. Usually this is the concatenation of a directory-level flag and target-specific flag. The sed generator also tells kmake about a `SED_SCRIPT` flag that it should record. kmake will automatically notice directory-level `SED_SCRIPT`-y vars as well as target-specific target-`SED_SCRIPT`-y vars and store them. The generator then uses the value as a script that is passed to sed. The value of the `SED_SCRIPT` flag is also fed into the `$(CMD)` variable, i.e. kmake will populate the .cmd file with the script. As a result, changing the script will automatically force regeneration of output files. Generator recipe ```````````````` In the the recipe part the generator must define a plain make recipe to actually generate the output files. Usually it's best to take the list of targets and generate a recipe for each as done in the sed generator. But it may be necessary to define a pattern rule, especially if the generator outputs more than one file. Here kmake cannot aid much. You can reference target specific variables that are setup in the rule part of the generator to pass options to the generating program. Additionally, you can use kmake's `printcmd` macro to output a pretty line in the absence of ``make V=1`` (see `Silent / verbose output`_).
About
kbuild-inspired build system to replace Automake, using only GNU Make
Resources
Stars
Watchers
Forks
Releases
No releases published
Packages 0
No packages published