Contents
Miscellaneous
- Whitespacing is significant in Make:
- All commands must be indented one tab.
- Spaces are not the same as tabs.
- Targets are not indented.
- ../Makefile.inc is automatically included if you include any bsd.*.mk, but never ../../Makefile.inc or ./Makefile.inc, unless you include these explicitly of course.
- You can include files optionally with .-include "filename". You won't get an error if the file doesn't exist. .sinclude is an alias for .-include.
- always list non-file targets in a .PHONY:-line. If there ever will be a file of the same name as your phony targets, make would otherwise immediately stop claiming that 'foo is up to date'. Also, if someone updates modification and creation times with make -t, the 'file' won't get created.
Installing neatly to a user-customisable path can be done most cleanly like this:
PREFIX?= /usr/local BINDIR= ${PREFIX}/bin LIBDIR= ${PREFIX}/lib MANDIR= ${PREFIX}/man FILESDIR= ${PREFIX}/share/misc
Added advantage is that this will work out-of-the-box on pkgsrc and FreeBSD's ports, since these define PREFIX to be their installation paths.
- Switch manpage generation/installation on and off with MKMAN, by setting it to "yes" or "no".
Make include files (types of projects)
bsd.prog.mk (programs)
Expects a manpage of the name ${PROG}.1, if MAN is not defined. Initialise MAN to an empty value if you do not have a manpage:
PROG=foo
SRCS=foo.c
MAN=
.include <bsd.prog.mk>
bsd.lib.mk (libraries)
Does not create shared libraries unless you define SHLIB_MAJOR variable:
LIB=this
SHLIB_MAJOR=1
SHLIB_MINOR=0
.include <bsd.lib.mk>
You may define SHLIB_TEENY variable too.
Alternatively, you may define variables in "shlib_version" file:
major=0
minor=1
where the 0 and 1 are replaced by the appropriate version numbers, of course.
bsd.man.mk (manpage stuff)
You can automatically have BSD Make create symlinks to your base manpages by defining MLINKS as a list of manpage.1 linkname.1 tuples separated by spaces. Example:
MLINKS = real_foo.3 foo.3 real_foo.3 other_foo.3
Here, real_foo.3 is a real file, the rest are all links that will be created in the installation path.
- Manpages must have an extension to identify their category, or installation will fail.*
Special constructs
Special constructs (.if, .for, .include etc) must have their dot flush left aligned. The word after the dot may be indented, however. You can check the current target being built by make with
.if make(targname)
blah
.endif
where targname is the target you're checking for.
When you check the contents of a variable, if it does not exist (it hasn't been defined), you will get 'strange' errors:
.if ${VAR} == "yes"
blah
.endif
will get you:
make: "/home/foo/src/libfoo" line 7: Malformed conditional (${VAR} == "yes")
make: "/home/foo/src/libfoo" line 7: Need an operator
make: "/home/foo/src/libfoo" line 9: if-less endif
make: "/home/foo/src/libfoo" line 9: Need an operator
make: Fatal errors encountered -- cannot continue
This kind of error is quite subtle since it might never occur on your system because you happen to always have the variable defined. To fix this, enclose the tested variable in quotes:
.if "${VAR}" == "yes"
blah
.endif
(The first example evaluates to .if == "yes" while the second to something slightly different: .if "" == "yes")
See also
- Basic Unix programming
- http://cvsweb.netbsd.org/bsdweb.cgi/src/usr.bin/make/?only_with_tag=MAIN
- http://cvsweb.netbsd.org/bsdweb.cgi/src/share/mk/?only_with_tag=MAIN
- http://www.crufty.net/help/sjg/bmake.htm
- http://www.crufty.net/help/sjg/mk-files.htm