manpagez: man pages & more
info make
Home | html | info | man

File: make.info,  Node: Simple Assignment,  Next: Immediate Assignment,  Prev: Recursive Assignment,  Up: Flavors

6.2.2 Simply Expanded Variable Assignment
-----------------------------------------

To avoid the problems and inconveniences of recursively expanded
variables, there is another flavor: simply expanded variables.

   "Simply expanded variables" are defined by lines using ':=' or '::='
(*note Setting Variables: Setting.).  Both forms are equivalent in GNU
'make'; however only the '::=' form is described by the POSIX standard
(support for '::=' is added to the POSIX standard for POSIX Issue 8).

   The value of a simply expanded variable is scanned once, expanding
any references to other variables and functions, when the variable is
defined.  Once that expansion is complete the value of the variable is
never expanded again: when the variable is used the value is copied
verbatim as the expansion.  If the value contained variable references
the result of the expansion will contain their values _as of the time
this variable was defined_.  Therefore,

     x := foo
     y := $(x) bar
     x := later

is equivalent to

     y := foo bar
     x := later

   Here is a somewhat more complicated example, illustrating the use of
':=' in conjunction with the 'shell' function.  (*Note The 'shell'
Function: Shell Function.)  This example also shows use of the variable
'MAKELEVEL', which is changed when it is passed down from level to
level.  (*Note Communicating Variables to a Sub-'make':
Variables/Recursion, for information about 'MAKELEVEL'.)

     ifeq (0,${MAKELEVEL})
     whoami    := $(shell whoami)
     host-type := $(shell arch)
     MAKE := ${MAKE} host-type=${host-type} whoami=${whoami}
     endif

An advantage of this use of ':=' is that a typical 'descend into a
directory' recipe then looks like this:

     ${subdirs}:
             ${MAKE} -C $@ all

   Simply expanded variables generally make complicated makefile
programming more predictable because they work like variables in most
programming languages.  They allow you to redefine a variable using its
own value (or its value processed in some way by one of the expansion
functions) and to use the expansion functions much more efficiently
(*note Functions for Transforming Text: Functions.).

   You can also use them to introduce controlled leading whitespace into
variable values.  Leading whitespace characters are discarded from your
input before substitution of variable references and function calls;
this means you can include leading spaces in a variable value by
protecting them with variable references, like this:

     nullstring :=
     space := $(nullstring) # end of the line

Here the value of the variable 'space' is precisely one space.  The
comment '# end of the line' is included here just for clarity.  Since
trailing space characters are _not_ stripped from variable values, just
a space at the end of the line would have the same effect (but be rather
hard to read).  If you put whitespace at the end of a variable value, it
is a good idea to put a comment like that at the end of the line to make
your intent clear.  Conversely, if you do _not_ want any whitespace
characters at the end of your variable value, you must remember not to
put a random comment on the end of the line after some whitespace, such
as this:

     dir := /foo/bar    # directory to put the frobs in

Here the value of the variable 'dir' is '/foo/bar    ' (with four
trailing spaces), which was probably not the intention.  (Imagine
something like '$(dir)/file' with this definition!)

© manpagez.com 2000-2025
Individual documents may contain additional copyright information.