書き方
特殊変数の取り扱い
$@ : ターゲットファイル名
$% : ターゲットがアーカイブメンバだったときのターゲットメンバ名
$< : 最初の依存するファイルの名前
$? : ターゲットより新しいすべての依存するファイル名
$^ : すべての依存するファイルの名前
$+ : Makefileと同じ順番の依存するファイルの名前
$* : サフィックスを除いたターゲットの名前
イコールの違い
右辺の内容を憶えておき実際に使う時に展開される。代入には「=」を使う
代入行がMakefileから読みこまれるとすぐに右辺を評価する。
すべてのC,C++をコンパイル
$(patsubst %.cpp,%.o,$(wildcard .cpp)) $(patsubst %.c,%.o,$(wildcard .c))
使い方
LIBSやCFLAGSなどのシンボル
ldというリンクを呼び出すコマンドのリンクオプション
includeするheaderのディレクトリを指定する
ターゲットファイルを生成するために利用するソースコード
ターゲットファイルを生成するために利用するオブジェクトファイル
CFLAGS,LDFLAGSのオプションがどれか調べる
gccのリンカはldを呼んでいるだけなので、ldのオプションがLDFLAGSになる。
セキュアなCFLAGS,LDFLAGS
ライブラリ
CFLAGS=-fstack-protector-all -O2 -fno-strict-aliasing -D_FORTIFY_SOURCE=2
LDFLAGS=-Wl,-z,now,-z,relro
実行ファイルは追加で
CFLAGS=-fPIE
LDFLAGS=-pie
多段Make
べからず集
suffix ruleについて
suffix ruleは古い書き方で、見た目にも分かりにくい(依存関係の順番がpattern ruleと逆になっている)。GNU makeを使う限りにおいては、suffix ruleではなくpattern ruleを使う方がよいらしい。
PHONY
.PHONY: all
.PHONY: clean
は必ず書くこと。
targetがallやcleanでも誤動作しないようにするため
イコールの使い方
=は動作が直感的でないため:=を使用すること
ヘッダファイルの依存関係を解決すること
FAQ
make で同じターゲット名を定義
二重コロンでターゲットを定義すると重複を許容できる
==> 1.mk <==
all ::
echo 1
==> 2.mk <==
all ::
echo 2
==> 3.mk <==
all ::
echo 3
includeするときの ディレクトリの場所が移動できない
Makefileから別のMakefileを呼ぶ場合、カレントディレクトリから移動できない
次の変数を定義して使用すればよい。
TOP := $(dir $(lastword $(MAKEFILE_LIST)))
or
ROOT_DIR:=$(shell dirname $(realpath $(lastword $(MAKEFILE_LIST))))
or
mkfile_path := $(abspath $(lastword $(MAKEFILE_LIST)))
current_dir := $(notdir $(patsubst %/,%,$(dir $(mkfile_path))))
コマンドラインから引数を入力されても上書きさせず、追加させる方法
override CFLAGS +=
とする
エラーで止まらないようにするには
コマンドの先頭に-をつければよい。
ヘッダーファイルの依存関係を解決する方法
にやり方が書いてある
gcc -MMDを使う方法と-include を使う方法
参考
サンプル
これがなかなかいいサンプル
PROGNAME := $(shell basename `readlink -f ..`)
PRODIR := bin
OBJDIR := lib
INCDIR := include
SRCS := $(wildcard *.c)
OBJS := $(SRCS:%.c=%.o)
OBJSPATH := $(addprefix $(OBJDIR)/, $(OBJS))
DEPENDS := $(SRCS:%.c=%.d)
DEPENDSPATH := $(addprefix $(OBJDIR)/, $(DEPENDS))
INCL := -I$(INCDIR)/cspice/ -I$(INCDIR)/etc
LIBS := -ansi -O2 -fPIC -DNON_UNIX_STDIO -lm cspice.a csupport.a
CC := gcc
override CFLAGS +=
LD := gcc
override LDFLAGS +=
.PHONY: all
all:$(PROGNAME)
-include $(DEPENDSPATH)
$(PROGNAME): $(OBJSPATH)
$(LD) $(LDFLAGS) $^ $(LIBS) -o $(PRODIR)/$(PROGNAME)
$(OBJDIR)/%.o: %.c
$(CC) -c $(CFLAGS) $(INCL) $< -o $@
.PHONY: clean
clean:
rm -f $(OBJSPATH)
rm -f $(DEPENDSPATH)
rm -f $(PRODIR)/$(PROGNAME)
# google test
TEST_PACKAGE = $(PROGNAME)_test
TEST_SRC = $(PROGNAME)_test.cc
TEST_PRODIR = ../test/bin
TEST_OBJDIR = ../test/lib
TEST_SUDIR = ../test
TEST_DATADIR = ../testdata
TEST_OBJ = $(TEST_SRC:.cc=.o)
TEST_EXE = $(TEST_SRC:.cc=)
TEST_EXEPATH = $(addprefix $(TEST_PRODIR)/,$(TEST_EXE))
TEST_OBJPATH = $(addprefix $(TEST_OBJDIR)/,$(TEST_OBJ))
GTEST_DIR = /usr/local/gtest
GTEST_LIBS = $(GTEST_DIR)/lib/libgtest_main.a $(GTEST_DIR)/lib/libgtest.a
GTEST_HEADERS = $(GTEST_DIR)/include
GCPPFLAGS = -I../src -I$(GTEST_HEADERS)
GLDFLAGS = -lpthread
MAKESHELL = $(addsuffix ;, $(TEST_EXEPATH))
### google test ###
test: $(TEST_EXEPATH)
$(TEST_PRODIR)/%: $(TEST_OBJDIR)/%.o
g++ $(CFLAGS) -o $@ $^ $(LIBS) $(GTEST_LIBS) $(INCL) $(GLDFLAGS)
$(TEST_OBJDIR)/%.o: $(TEST_SUDIR)/%.cc
こっちもいい
TARGET ?= a.out
SRC_DIRS ?= ./src
SRCS := $(shell find $(SRC_DIRS) -name *.cpp -or -name *.c -or -name *.s)
OBJS := $(addsuffix .o,$(basename $(SRCS)))
DEPS := $(OBJS:.o=.d)
INC_DIRS := $(shell find $(SRC_DIRS) -type d)
INC_FLAGS := $(addprefix -I,$(INC_DIRS))
CPPFLAGS ?= $(INC_FLAGS) -MMD -MP
$(TARGET): $(OBJS)
$(CC) $(LDFLAGS) $(OBJS) -o $@ $(LOADLIBES) $(LDLIBS)
.PHONY: clean
clean:
$(RM) $(TARGET) $(OBJS) $(DEPS)
-include $(DEPS)