-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathMakefile.common.footer
474 lines (388 loc) · 18.1 KB
/
Makefile.common.footer
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
# -*- Makefile -*-
# This is a part of the mrbuild project: https://github.com/dkogan/mrbuild
#
# Released under an MIT-style license. Modify and distribute as you like:
#
# Copyright 2016-2019 California Institute of Technology
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in all
# copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
# SOFTWARE.
# This is a common Makefile that can be used as the core buildsystem for
# projects providing a library and some executables using this library. Please
# see README.build.org for the documentation.
# There are two ways to pass variables to make:
#
# make CFLAGS=-foo
# and
# CFLAGS=-foo make
#
# The former creates a "command line" variable and the latter an
# "environment variable". In order to be able to modify a "command line"
# variable (to add other flags, say), one MUST use 'override'. So one would have to do
#
# override CFLAGS += -bar
#
# without the "override" nothing would happen. I want to avoid this rabbithole
# entirely, so I disallow "command line" variables for things that I modify.
#
# I only do this for xxxFLAGS becuase cross-building and installing in Debian
# uses Make in this way. Hopefully this is safe enough
$(foreach v,$(filter %FLAGS,$(.VARIABLES)),$(if $(filter command line,$(origin $v)), $(error '$v' not allowed as a make parameter. Please do "$v=xxxx make yyyy" instead of "make yyyy $v=xxxx")))
# Make sure I have the variables that must be defined. Libraries need an ABI and
# TAIL version, while other need a plain VERSION
MUST_DEF_VARIABLES := PROJECT_NAME $(if $(LIB_SOURCES),ABI_VERSION TAIL_VERSION,VERSION)
$(foreach v,$(MUST_DEF_VARIABLES),$(if $($v),,$(error User MUST specify $v)))
# The default VERSION string that appears as a #define to each source file, and
# to any generated documentation (gengetopt and so on). The user can set this to
# whatever they like
VERSION ?= $(ABI_VERSION).$(TAIL_VERSION)
# Default compilers. By default, we use g++ as a linker
CC ?= gcc
CXX ?= g++
NVCC ?= nvcc
CC_LINKER ?= $(CXX)
# used to make gcc output header dependency information. All source
# files generated .d dependency definitions that are included at the
# bottom of this file
CCXXFLAGS += -MMD -MP
# always building with debug information. This is stripped into the
# -dbg/-debuginfo packages by debhelper/rpm later
CCXXFLAGS += -g
# I want the frame pointer. Makes walking the stack WAY easier
CCXXFLAGS += -fno-omit-frame-pointer
LIB_OBJECTS := $(addsuffix .o,$(basename $(LIB_SOURCES)))
BIN_OBJECTS := $(addsuffix .o,$(basename $(BIN_SOURCES)))
SOURCE_DIRS := $(sort ./ $(dir $(LIB_SOURCES) $(BIN_SOURCES)))
# if the PROJECT_NAME is libxxx then LIB_NAME is libxxx
# if the PROJECT_NAME is xxx then LIB_NAME is libxxx
LIB_NAME := $(or $(filter lib%,$(PROJECT_NAME)),lib$(PROJECT_NAME))
LIB_TARGET_SO_BARE := $(LIB_NAME).$(SO)
LIB_TARGET_SO_ABI := $(LIB_TARGET_SO_BARE).$(ABI_VERSION)
LIB_TARGET_SO_FULL := $(LIB_TARGET_SO_ABI).$(TAIL_VERSION)
LIB_TARGET_SO_ALL := $(LIB_TARGET_SO_BARE) $(LIB_TARGET_SO_ABI) $(LIB_TARGET_SO_FULL)
BIN_TARGETS := $(basename $(BIN_SOURCES))
# all objects built for inclusion in shared libraries get -fPIC. We don't build
# static libraries, so this is 100% correct
$(LIB_OBJECTS): CCXXFLAGS += -fPIC
CCXXFLAGS += -DMRBUILD_VERSION='"$(VERSION)"'
# These are here to process the options separately for each file being built.
# This allows per-target options to be set
#
# if DEB_BUILD_OPTIONS has "noopt" (we're building a debian package, and the
# user requested no optimizations), I remove all -O and add -O0
#
# otherwise, if no explicit optimization flags are given, I pass -O3
define massageopts
$(if $(filter noopt,$(DEB_BUILD_OPTIONS)),$(filter-out -O%,$1) -O0,\
$1 $(if $(filter -O%,$1),,-O3))
endef
# If no C++ standard requested, I default to c++0x
define massageopts_cxx
$(call massageopts,$1 $(if $(filter -std=%,$1),,-std=c++0x))
endef
define massageopts_c
$(call massageopts,$1)
endef
# define the compile rules. I need to redefine the rules here because my
# C..FLAGS variables are simple (immediately evaluated), but the user
# could have specified per-target flags that ALWAYS evaluate deferred-ly
#
# I add the warning options AT THE START of the flag list so that the user can
# override these
cc_build_rule = $(strip $(CXX) $(call massageopts_cxx,-Wall -Wextra $(CXXFLAGS) $(CCXXFLAGS) $(CPPFLAGS))) -c -o $@ $<
c_build_rule = $(strip $(CC) $(call massageopts_c, -Wall -Wextra $(CFLAGS) $(CCXXFLAGS) $(CPPFLAGS))) -c -o $@ $<
cu_build_rule = $(strip $(NVCC) $(call massageopts_c, -Wall -Wextra $(CUFLAGS) $(CPPFLAGS))) -c -o $@ $<
cu_build_rule += --compiler-options "-Wall -Wextra $(CCXXFLAGS)"
%.o:%.C
$(cc_build_rule)
%.o:%.cc
$(cc_build_rule)
%.o:%.cpp
$(cc_build_rule)
%.o:%.c
$(c_build_rule)
%.o:%.cu
$(cu_build_rule)
%.o: %.S
$(CC) $(ASFLAGS) $(CPPFLAGS) -c -o $@ $<
# gengetopt rule. If we have GENGETOPT_OPTIONS, use those; otherwise use some
# sensible defaults. If we don't have -F set an --output-dir also
%.h %.c: %.ggo
gengetopt -i $< \
$(if $(GENGETOPT_OPTIONS),$(if $(filter -F%,$(GENGETOPT_OPTIONS)),,--output-dir $(dir $<))) \
$(or $(GENGETOPT_OPTIONS), -C -u -g $(VERSION) -F $* args -f $(notdir $*) -a $(notdir $*))
# this is how you build QT4 UIs. There's a tool to generate headers from
# UI definitions. There's also a tool to generate metadata from QT
# classes. This must be compiled and linked in. QT4_MOCS is a list of
# all these extra objects needed by QT. Simply add $(QT4_MOCS) to your
# objects list and magic happens. Similar with QT4_UI_HEADERS
ui_%.h: %.ui
uic-qt4 $< > $@
moc_%.cpp: %.h
moc-qt4 $< > $@
ui_%.hh: %.ui
uic-qt4 $< > $@
moc_%.cpp: %.hh
moc-qt4 $< > $@
# Python docstring rules. I construct these from plain ASCII files to handle
# line wrapping
%.docstring.h: %.docstring
< $^ sed 's/\\/\\\\/g; s/"/\\"/g; s/^/"/; s/$$/\\n"/;' > $@
EXTRA_CLEAN += *.docstring.h
# by default I build shared libraries only. We known how to build static
# libraries too, but I don't do it unless asked
all: $(if $(strip $(LIB_SOURCES)),$(LIB_TARGET_SO_ALL)) $(if $(strip $(BIN_SOURCES)),$(BIN_TARGETS))
.PHONY: all
.DEFAULT_GOAL := all
ifeq ($(COND_DARWIN),)
$(LIB_TARGET_SO_FULL): LDFLAGS += -Wl,--default-symver
else
$(LIB_TARGET_SO_FULL): LDFLAGS += -dynamiclib
endif
$(LIB_TARGET_SO_FULL): LDFLAGS += -shared -fPIC -Wl,$(if $(COND_DARWIN),-install_name,-soname),$(notdir $(LIB_TARGET_SO_BARE)).$(ABI_VERSION)
$(LIB_TARGET_SO_BARE) $(LIB_TARGET_SO_ABI): $(LIB_TARGET_SO_FULL)
ln -fs $(notdir $(LIB_TARGET_SO_FULL)) $@
# Here instead of specifying $^, I do just the %.o parts and then the
# others. This is required to make the linker happy to see the dependent
# objects first and the dependency objects last. Same as for BIN_TARGETS
$(LIB_TARGET_SO_FULL): $(LIB_OBJECTS)
$(CC_LINKER) $(LDFLAGS) $(filter %.o, $^) $(filter-out %.o, $^) $(LDLIBS) -o $@
# I make sure to give the .o to the linker before the .so and everything else.
# The .o may depend on the other stuff.
#
# The binaries get an RPATH (removed at install time). I use a relative RPATH
# (using $ORIGIN) to make the build reproducible: chrpath doesn't remove the
# actual RPATH string from the binary, and the local build directory gets
# embedded in the output
#
# Here $^ contains two flavors of LIB_TARGET (see next stanza), so I manually
# remove one
$(BIN_TARGETS): %: %.o
$(CC_LINKER) $(RPATH_OPTION) $(LDFLAGS) $(filter %.o, $^) $(filter-out $(LIB_TARGET_SO_ABI),$(filter-out %.o, $^)) $(LDLIBS) -o $@
# The binaries link with the DSO, if there is one. I need the libxxx.so to build
# the binary, and I need the libxxx.so.abi to run it.
$(BIN_TARGETS): $(if $(strip $(LIB_SOURCES)),$(LIB_TARGET_SO_BARE) $(LIB_TARGET_SO_ABI))
%.$(SO).dump: %.$(SO)
abi-dumper $< -o $@
EXTRA_CLEAN += *.$(SO).dump
clean:
rm -rf $(foreach d,$(SOURCE_DIRS),$(addprefix $d,*.a *.o *.$(SO) *.$(SO).* *.d moc_* ui_*.h*)) $(BIN_TARGETS) $(foreach s,.c .h,$(addsuffix $s,$(basename $(shell find . -name '*.ggo')))) $(EXTRA_CLEAN)
distclean: clean
.PHONY: distclean clean
########################### installation business
ifneq (,$(filter install,$(MAKECMDGOALS)))
ifeq ($(strip $(DESTDIR)),)
$(error Tried to make install without having DESTDIR defined \
"make install" is ONLY for package building. \
What are you trying to do?)
endif
ifneq (,$(strip $(DIST_PY2_MODULES)))
PY2_MODULE_PATH := $(shell python2 -c "from distutils.sysconfig import get_python_lib; print(get_python_lib())")
$(if $(PY2_MODULE_PATH),,$(error "Couldn't find the python2 module path!"))
endif
ifneq (,$(strip $(DIST_PY3_MODULES)))
PY3_MODULE_PATH := $(shell python3 -c "from distutils.sysconfig import get_python_lib; print(get_python_lib())")
$(if $(PY3_MODULE_PATH),,$(error "Couldn't find the python3 module path!"))
endif
USE_DEBIAN_PATHS := $(wildcard /etc/debian_version)
ifneq (,$(USE_DEBIAN_PATHS))
# we're a debian-ish box, use the multiarch dir
DEB_HOST_MULTIARCH := $(shell dpkg-architecture -qDEB_HOST_MULTIARCH 2>/dev/null)
USRLIB := usr/lib/$(DEB_HOST_MULTIARCH)
else
# we're something else. Do what CentOS does.
# If /usr/lib64 exists, use that. Otherwise /usr/lib
USRLIB := $(if $(wildcard /usr/lib64),usr/lib64,usr/lib)
endif
endif
# I process the simple wildcard exceptions on DIST_BIN and DIST_INCLUDE in a
# deferred fashion. The reason is that I need $(wildcard) to run at install
# time, i.e. after stuff is built, and the files $(wildcard) is looking at
# already exist
DIST_BIN_ORIG := $(DIST_BIN)
DIST_INCLUDE_ORIG := $(DIST_INCLUDE)
DIST_BIN = $(filter-out $(wildcard $(DIST_BIN_EXCEPT)), \
$(wildcard $(or $(DIST_BIN_ORIG), $(BIN_TARGETS))))
DIST_INCLUDE = $(filter-out $(wildcard $(DIST_INCLUDE_EXCEPT) *.docstring.h), \
$(wildcard $(DIST_INCLUDE_ORIG)))
MANDIR := usr/share/man
# Generates the install rules. Arguments:
# 1. variable containing the being installed
# 2. target path they're being installed to
define install_rule
$(if $(strip $($1)),
mkdir -p $2 && \
cp -r $($1) $2 && \
$(if $($(1)_EXCEPT_FINDSPEC),find $2 $($(1)_EXCEPT_FINDSPEC) -delete, true) )
endef
# Redhat wants the various manpage section get their own subdir for some reason
define move_manpages_for_redhat
mkdir -p $(DESTDIR)/$(MANDIR)/man$1 && \
(mv $(DESTDIR)/$(MANDIR)/*.$1 $(DESTDIR)/$(MANDIR)/man$1 2>/dev/null || true) && \
(mv $(DESTDIR)/$(MANDIR)/*.$1.gz $(DESTDIR)/$(MANDIR)/man$1 2>/dev/null || true) && \
(rmdir $(DESTDIR)/$(MANDIR)/man$1 2>/dev/null || true)
endef
ifneq ($(strip $(LIB_SOURCES)),)
install: $(LIB_TARGET_SO_ALL)
endif
install: $(BIN_TARGETS) $(DIST_DOC) $(DIST_MAN) $(DIST_DATA)
# using 'cp -P' instead of 'install' because the latter follows links unconditionally
ifneq ($(strip $(LIB_SOURCES)),)
mkdir -p $(DESTDIR)/$(USRLIB)
cp -P $(LIB_TARGET_SO_FULL) $(DESTDIR)/$(USRLIB)
ln -fs $(notdir $(LIB_TARGET_SO_FULL)) $(DESTDIR)/$(USRLIB)/$(notdir $(LIB_TARGET_SO_ABI))
ln -fs $(notdir $(LIB_TARGET_SO_FULL)) $(DESTDIR)/$(USRLIB)/$(notdir $(LIB_TARGET_SO_BARE))
endif
$(call install_rule,DIST_BIN, $(DESTDIR)/usr/bin)
$(call install_rule,DIST_INCLUDE, $(DESTDIR)/usr/include/$(PROJECT_NAME))
$(call install_rule,DIST_DOC, $(DESTDIR)/usr/share/doc/$(PROJECT_NAME))
# I install the manpages normally. On Redhat I need to shuffle them into
# different subdirectories. This is incomplete: sections that aren't simply
# digits 1-9 will not be moved. "3perl" is an example I can think of that won't
# work here. Good-enough for now
$(call install_rule,DIST_MAN, $(DESTDIR)/$(MANDIR))
ifeq (,$(USE_DEBIAN_PATHS))
$(foreach s,1 2 3 4 5 6 7 8 9,$(call move_manpages_for_redhat,$s) && ) true
endif
$(call install_rule,DIST_DATA, $(DESTDIR)/usr/share/$(PROJECT_NAME))
$(call install_rule,DIST_PERL_MODULES,$(DESTDIR)/usr/share/perl5)
$(call install_rule,DIST_PY2_MODULES, $(DESTDIR)/$(PY2_MODULE_PATH))
$(call install_rule,DIST_PY3_MODULES, $(DESTDIR)/$(PY3_MODULE_PATH))
# In filenames I rename __colon__ -> :
# This is required because Make can't deal with : in rules
for fil in `find $(DESTDIR) -name '*__colon__*'`; do mv $$fil `echo $$fil | sed s/__colon__/:/g`; done
ifeq ($(COND_DARWIN),)
# Remove rpaths from everything. /usr/bin is allowed to fail because
# some of those executables aren't ELFs. On the other hand, any .so we
# find IS en ELF. Those could live in a number of places, since they
# could be extension modules for the various languages, and I thus look
# for those EVERYWHERE
ifneq ($(strip $(DIST_BIN)),)
chrpath -d $(DESTDIR)/usr/bin/* 2>/dev/null || true
endif
find $(DESTDIR) -name '*.$(SO)' | xargs chrpath -d
else
# BSD. I need to strip out the RPATHs individually
ifneq ($(strip $(DIST_BIN)),)
# here $f is the built file
$(foreach f,$(filter $(DIST_BIN),$(BIN_TARGETS)),install_name_tool -delete_rpath $(call RPATH,$f) $(DESTDIR)/usr/bin/$(notdir $f) && \$(NEWLINE)) true
endif
echo "bsd install-time stripping of rpath is incomplete:"
echo "1. need py2 path"
echo "2. does $(wildcard) work here? it will be expanded before those targets exist"
echo "3. what about nested paths? test mrgingham"
ifneq ($(strip $(DIST_PY3_MODULES)),)
# here $f is the installed file
$(foreach f,$(filter %$(PY_EXT_SUFFIX),$(wildcard $(DESTDIR)/$(PY3_MODULE_PATH)/$(DIST_PY3_MODULES)/*)),install_name_tool -delete_rpath $(call RPATH,$(patsubst $(DESTDIR)/$(PY3_MODULE_PATH)/%,%,$f)) $f && \$(NEWLINE)) true
endif
endif
# Any perl programs need their binary path stuff stripped out. This
# exists to let these run in-tree, but needs to be removed at
# install-time (similar to an RPATH)
ifneq ($(strip $(DIST_BIN)),)
for fil in `find $(DESTDIR)/usr/bin -type f`; do head -n1 $$fil | grep -q '^#!.*/perl$$' && perl -n -i -e 'print unless /^\s* use \s+ lib \b/x' $$fil || true; done
endif
test -e $(DESTDIR)/usr/share/perl5 && find $(DESTDIR)/usr/share/perl5 -type f | xargs perl -n -i -e 'print unless /^\s* use \s+ lib \b/x' || true
ifneq ($(strip $(DIST_BIN)),)
# Everything executable needs specific permission bits
chmod 0755 $(DESTDIR)/usr/bin/*
endif
.PHONY: install
# I don't want any intermediate files. There's a good summary of this situation
# by Masahiro Yamada here:
#
# https://lore.kernel.org/lkml/[email protected]/T/
#
# And this snippet of code is from his patch (except I exclude versions 4.4...
# to support the "remake" tool).
#
# I don't want any intermediate files because:
# - they are deleted after each build
# - a missing intermediate file will NOT trigger a rebuild of targets that
# depend on it
#
# A demo. Create this Makefile:
#
# # Marking all files intermediate using ".SECONDARY:" breaks this: a missing "b"
# # doesn't trigger a rebuild of "a"
# a: b
# touch $@
# b:
# touch $@
#
# # A common chain of build steps. Without .SECONDARY some of these will be
# # deleted after the initial build, and will be re-created in a subsequent build
# %-GENERATED.c: %-generate
# touch $@
# %.o: %.c
# touch $@
# %.so: %-GENERATED.o
# touch $@
# xxx-GENERATED.o: CFLAGS += adsf
#
# # Imitates .d files created with "gcc -MMD". Does not exist on the initial build
# # (so the xxx-GENERATED.[co] files ARE seen as intermediate initially). But DOES
# # exist on subsequence builds, so these files are NOT seen as intermediate
# ifneq ($(wildcard xxx.so),)
# xxx-GENERATED.o: xxx-GENERATED.c
# endif
#
# clean:
# rm -rf a b xxx-GENERATED.c xxx-GENERATED.o xxx.so
# .PHONY: clean
#
# Then run
#
# make clean
# touch a xxx-generate xxx.h
# echo "\n*** Should build xxx stuff AND a,b; should NOT rm xxx-GENERATED.c"
# make a xxx.so
# echo "\n*** All built should do nothing"
# make a xxx.so
#
# This
# - creates the xxx... files that would exist in a source tree
# - sets us with with "a" but not "b", so we SHOULD rebuild "a"
#
# So we should see the initial build create everything, and not delete anything;
# and the subsequent build shoulnd't have to do anything. And we can see that it
# doesn't do the right thing with or without .SECONDARY:
#
# The solution is to not have any intermediate files. We can mark everything as
# not-intermediate with .NOTINTERMEDIATE:, but that's only available in make
# 4.4, and apparently was buggy in the initial release. We can also explicitly
# mention specific files as targets or prerequisites (for instance "___dummy___:
# file1 file 2") but that can only be done by the specific Makefiles using
# mrbuild, not mrbuild itself. So if we cannot have .NOTINTERMEDIATE: here, we
# settle for .SECONDARY: this won't delete stuff after each build, but still has
# the second problem: missing files will NOT trigger a rebuild.
#
# Masahiro Yamada's snippet:
#
# .NOTINTERMEDIATE is more correct, but only available on newer Make versions.
# Make 4.4 introduced .NOTINTERMEDIATE, and it appears in .FEATURES, but the
# global .NOTINTERMEDIATE does not work. We can use it on Make > 4.4.
# Use .SECONDARY for older Make versions, but "newer-prereq" cannot detect
# deleted files.
ifneq ($(and $(filter notintermediate, $(.FEATURES)),$(filter-out 4.4%,$(MAKE_VERSION))),)
.NOTINTERMEDIATE:
else
.SECONDARY:
endif
# the header dependencies
-include $(addsuffix *.d,$(SOURCE_DIRS))