Variables
Make supports variable assignment. All variables in make are strings:
my_variable = this is the content
You can then reference the text in the variable by using $(VAR)
:
my_variable = this is the content
default:
echo "$(my_variable)"
Variables just one character long may omit the
()
wrapping the variable name. E.g.$(a)
is the same as$a
.
To use the variable, make will literally remove $(VAR)
and put the string
inside the variable in its place.
This is important to remember.
Make literally substitutes the string in place of the variable, without
any magic.
The above file is exactly the same as this one:
default:
echo: "this is the content"
Make first reads the whole makefile, then substitutes the variables, and then executes the rules.
You will have noticed that the
$(VAR)
syntax is also used for shell variable substitutions and function calls. If you want to use shell variables in your rules, you will need to escape the$
used by make by doubling it; e.g. use$$(SHELL_VAR)
instead of$(VAR)
. While expanding variables, make will convert$$
to$
, and pass it to the shell, therefore making the canonical$
for shell invocations.However, you can use shell variables directly with
$()
: continue reading!
Make copies all shell environment variables upon starting as if they where written in the makefile. For example, this works:
path_content.txt:
# Notice that the variable here is a make variable, since it has just
# one \$ not two.
echo $(PATH) > path_content.txt
You can conditionally override environment variables with the ?=
assignment.
This assigns the value of the variable only if it's not already assigned:
some_var ?= my_text
default:
echo $(some_var)
If you run make
, the output will be "my text"
.
If you run some_var="alternative text" make
, the output will be "alternative text"
,
since the assignment in the makefile will not be made.
Try to keep your makefile variables
snake_case
.
What to do with variables
You can do a lot with variables. They are most commonly used to write the requirements for a rule:
files = one.txt two.txt three.txt
output: $(files)
...
You can also use them to shorten long calls:
e_dir = long/path/to/executable/directory
flags = --some --default --flags --that-are --always-used
output:
$(e_dir)/create_file $(flags) > output
This is very useful to work with stuff like
Rscript
:r = Rscript --vanilla output: $(r) my_r_script.R > output
You can use conditional assignment for variables that the user can override:
option ?= default_value
default:
execute --var $(option)
The user can use the default value or alternatively export an option
variable
and override it.
This can be useful when debugging makefiles: the default option is what you
want to use for a "regular" run, but you can override it during development to
get, e.g. debug information.
Automatic variables
When a recipe is run, make sets some automatic variables, so that you can write your recipes in a less verbose way.
You can read the full list of all automatic variables in the make manual.
Here are some of the most commonly used ones:
$@
is the target of the rule:path/to/target.txt: requirement.txt cat requirement.txt > $@ # cat requirement.txt > path/to/target.txt
$<
is the first requirement. Careful when using this when you have more than one requirement, as it's order specific:path/to/target.txt: requirement.txt cat $< > $@ # cat requirement.txt > path/to/target.txt
$(@D)
and$(@F)
are the directory of the target file and the name of the target file, respectively. This is very useful to create containing folders for output files:
You can do the same withpath/to/target.txt: requirement.txt mkdir -p $(@D) # mkdir -p path/to/ cat $< > $@ # cat requirement.txt > path/to/target.txt
$(<D)
and$(<F)
for the first requirement file.$^
is the full list of requirements, separated by spaces:target.txt: one.txt two.txt three.txt concat_files $^ # concat_files one.txt two.txt three.txt
You may wonder how to select the N-th requirement. There is no automatic variable for each requirement, but you can use
$^
to get it with theword
function:$(word n, $^)
wheren
is the 1-indexed position of the requirement you want. For example, ifone two three
are the requirements,$(word 2, $^)
will result in the stringtwo
.Read more about functions here or in the manual.