Makefile 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351
  1. ifdef BUILDDIR
  2. # make sure BUILDDIR ends with a slash
  3. override BUILDDIR := $(BUILDDIR)/
  4. # bit of a hack, but we want to make sure BUILDDIR directory structure
  5. # is correct before any commands
  6. $(if $(findstring n,$(MAKEFLAGS)),, $(shell mkdir -p \
  7. $(BUILDDIR) \
  8. $(BUILDDIR)bd \
  9. $(BUILDDIR)runners \
  10. $(BUILDDIR)tests \
  11. $(BUILDDIR)benches))
  12. endif
  13. # overridable target/src/tools/flags/etc
  14. ifneq ($(wildcard test.c main.c),)
  15. TARGET ?= $(BUILDDIR)lfs
  16. else
  17. TARGET ?= $(BUILDDIR)lfs.a
  18. endif
  19. CC ?= gcc
  20. AR ?= ar
  21. SIZE ?= size
  22. CTAGS ?= ctags
  23. NM ?= nm
  24. OBJDUMP ?= objdump
  25. VALGRIND ?= valgrind
  26. GDB ?= gdb
  27. PERF ?= perf
  28. SRC ?= $(filter-out $(wildcard *.*.c),$(wildcard *.c))
  29. OBJ := $(SRC:%.c=$(BUILDDIR)%.o)
  30. DEP := $(SRC:%.c=$(BUILDDIR)%.d)
  31. ASM := $(SRC:%.c=$(BUILDDIR)%.s)
  32. CI := $(SRC:%.c=$(BUILDDIR)%.ci)
  33. GCDA := $(SRC:%.c=$(BUILDDIR)%.t.a.gcda)
  34. TESTS ?= $(wildcard tests/*.toml)
  35. TEST_SRC ?= $(SRC) \
  36. $(filter-out $(wildcard bd/*.*.c),$(wildcard bd/*.c)) \
  37. runners/test_runner.c
  38. TEST_RUNNER ?= $(BUILDDIR)runners/test_runner
  39. TEST_TC := $(TESTS:%.toml=$(BUILDDIR)%.t.c) \
  40. $(TEST_SRC:%.c=$(BUILDDIR)%.t.c)
  41. TEST_TAC := $(TEST_TC:%.t.c=%.t.a.c)
  42. TEST_OBJ := $(TEST_TAC:%.t.a.c=%.t.a.o)
  43. TEST_DEP := $(TEST_TAC:%.t.a.c=%.t.a.d)
  44. TEST_CI := $(TEST_TAC:%.t.a.c=%.t.a.ci)
  45. TEST_GCNO := $(TEST_TAC:%.t.a.c=%.t.a.gcno)
  46. TEST_GCDA := $(TEST_TAC:%.t.a.c=%.t.a.gcda)
  47. TEST_PERF := $(TEST_RUNNER:%=%.perf)
  48. BENCHES ?= $(wildcard benches/*.toml)
  49. BENCH_SRC ?= $(SRC) \
  50. $(filter-out $(wildcard bd/*.*.c),$(wildcard bd/*.c)) \
  51. runners/bench_runner.c
  52. BENCH_RUNNER ?= $(BUILDDIR)runners/bench_runner
  53. BENCH_BC := $(BENCHES:%.toml=$(BUILDDIR)%.b.c) \
  54. $(BENCH_SRC:%.c=$(BUILDDIR)%.b.c)
  55. BENCH_BAC := $(BENCH_BC:%.b.c=%.b.a.c)
  56. BENCH_OBJ := $(BENCH_BAC:%.b.a.c=%.b.a.o)
  57. BENCH_DEP := $(BENCH_BAC:%.b.a.c=%.b.a.d)
  58. BENCH_CI := $(BENCH_BAC:%.b.a.c=%.b.a.ci)
  59. BENCH_GCNO := $(BENCH_BAC:%.b.a.c=%.b.a.gcno)
  60. BENCH_GCDA := $(BENCH_BAC:%.b.a.c=%.b.a.gcda)
  61. BENCH_PERF := $(BENCH_RUNNER:%=%.perf)
  62. ifdef DEBUG
  63. override CFLAGS += -O0
  64. else
  65. override CFLAGS += -Os
  66. endif
  67. ifdef TRACE
  68. override CFLAGS += -DLFS_YES_TRACE
  69. endif
  70. override CFLAGS += -g3
  71. override CFLAGS += -I.
  72. override CFLAGS += -std=c99 -Wall -pedantic
  73. override CFLAGS += -Wextra -Wshadow -Wjump-misses-init -Wundef
  74. override CFLAGS += -ftrack-macro-expansion=0
  75. ifdef YES_COVERAGE
  76. override CFLAGS += --coverage
  77. endif
  78. ifdef YES_PERF
  79. override CFLAGS += -fno-omit-frame-pointer
  80. endif
  81. ifdef VERBOSE
  82. override CODEFLAGS += -v
  83. override DATAFLAGS += -v
  84. override STACKFLAGS += -v
  85. override STRUCTFLAGS += -v
  86. override COVERAGEFLAGS += -v
  87. override PERFFLAGS += -v
  88. endif
  89. ifneq ($(NM),nm)
  90. override CODEFLAGS += --nm-tool="$(NM)"
  91. override DATAFLAGS += --nm-tool="$(NM)"
  92. endif
  93. ifneq ($(OBJDUMP),objdump)
  94. override CODEFLAGS += --objdump-tool="$(OBJDUMP)"
  95. override DATAFLAGS += --objdump-tool="$(OBJDUMP)"
  96. override STRUCTFLAGS += --objdump-tool="$(OBJDUMP)"
  97. override PERFFLAGS += --objdump-tool="$(OBJDUMP)"
  98. endif
  99. ifneq ($(PERF),perf)
  100. override PERFFLAGS += --perf-tool="$(PERF)"
  101. endif
  102. override TESTFLAGS += -b
  103. override BENCHFLAGS += -b
  104. # forward -j flag
  105. override TESTFLAGS += $(filter -j%,$(MAKEFLAGS))
  106. override BENCHFLAGS += $(filter -j%,$(MAKEFLAGS))
  107. ifdef YES_PERF
  108. override TESTFLAGS += --perf=$(TEST_PERF)
  109. endif
  110. ifndef NO_PERF
  111. override BENCHFLAGS += --perf=$(BENCH_PERF)
  112. endif
  113. ifdef VERBOSE
  114. override TESTFLAGS += -v
  115. override TESTCFLAGS += -v
  116. override BENCHFLAGS += -v
  117. override BENCHCFLAGS += -v
  118. endif
  119. ifdef EXEC
  120. override TESTFLAGS += --exec="$(EXEC)"
  121. override BENCHFLAGS += --exec="$(EXEC)"
  122. endif
  123. ifneq ($(GDB),gdb)
  124. override TESTFLAGS += --gdb-tool="$(GDB)"
  125. override BENCHFLAGS += --gdb-tool="$(GDB)"
  126. endif
  127. ifneq ($(VALGRIND),valgrind)
  128. override TESTFLAGS += --valgrind-tool="$(VALGRIND)"
  129. override BENCHFLAGS += --valgrind-tool="$(VALGRIND)"
  130. endif
  131. ifneq ($(PERF),perf)
  132. override TESTFLAGS += --perf-tool="$(PERF)"
  133. override BENCHFLAGS += --perf-tool="$(PERF)"
  134. endif
  135. # commands
  136. .PHONY: all build
  137. all build: $(TARGET)
  138. .PHONY: asm
  139. asm: $(ASM)
  140. .PHONY: size
  141. size: $(OBJ)
  142. $(SIZE) -t $^
  143. .PHONY: tags
  144. tags:
  145. $(CTAGS) --totals --c-types=+p $(shell find -H -name '*.h') $(SRC)
  146. .PHONY: test-runner build-test
  147. ifndef NO_COVERAGE
  148. test-runner build-test: override CFLAGS+=--coverage
  149. endif
  150. ifdef YES_PERF
  151. bench-runner build-bench: override CFLAGS+=-fno-omit-frame-pointer
  152. endif
  153. test-runner build-test: $(TEST_RUNNER)
  154. ifndef NO_COVERAGE
  155. rm -f $(TEST_GCDA)
  156. endif
  157. ifdef YES_PERF
  158. rm -f $(TEST_PERF)
  159. endif
  160. .PHONY: test
  161. test: test-runner
  162. ./scripts/test.py $(TEST_RUNNER) $(TESTFLAGS)
  163. .PHONY: test-list
  164. test-list: test-runner
  165. ./scripts/test.py $(TEST_RUNNER) $(TESTFLAGS) -l
  166. .PHONY: bench-runner build-bench
  167. ifdef YES_COVERAGE
  168. bench-runner build-bench: override CFLAGS+=--coverage
  169. endif
  170. ifndef NO_PERF
  171. bench-runner build-bench: override CFLAGS+=-fno-omit-frame-pointer
  172. endif
  173. bench-runner build-bench: $(BENCH_RUNNER)
  174. ifdef YES_COVERAGE
  175. rm -f $(BENCH_GCDA)
  176. endif
  177. ifndef NO_PERF
  178. rm -f $(BENCH_PERF)
  179. endif
  180. .PHONY: bench
  181. bench: bench-runner
  182. ./scripts/bench.py $(BENCH_RUNNER) $(BENCHFLAGS)
  183. .PHONY: bench-list
  184. bench-list: bench-runner
  185. ./scripts/bench.py $(BENCH_RUNNER) $(BENCHFLAGS) -l
  186. .PHONY: code
  187. code: $(OBJ)
  188. ./scripts/code.py $^ -Ssize $(CODEFLAGS)
  189. .PHONY: data
  190. data: $(OBJ)
  191. ./scripts/data.py $^ -Ssize $(DATAFLAGS)
  192. .PHONY: stack
  193. stack: $(CI)
  194. ./scripts/stack.py $^ -Slimit -Sframe $(STACKFLAGS)
  195. .PHONY: struct
  196. struct: $(OBJ)
  197. ./scripts/struct_.py $^ -Ssize $(STRUCTFLAGS)
  198. .PHONY: coverage
  199. coverage: $(GCDA)
  200. $(strip ./scripts/coverage.py \
  201. $^ $(patsubst %,-F%,$(SRC)) \
  202. -slines -sbranches \
  203. $(COVERAGEFLAGS))
  204. .PHONY: perf
  205. perf: $(BENCH_PERF)
  206. $(strip ./scripts/perf.py \
  207. $^ $(patsubst %,-F%,$(SRC)) \
  208. -Scycles \
  209. $(PERFFLAGS))
  210. .PHONY: summary sizes
  211. summary sizes: $(BUILDDIR)lfs.csv
  212. $(strip ./scripts/summary.py -Y $^ \
  213. -fcode=code_size \
  214. -fdata=data_size \
  215. -fstack=stack_limit \
  216. -fstruct=struct_size \
  217. --max=stack \
  218. $(SUMMARYFLAGS))
  219. # rules
  220. -include $(DEP)
  221. -include $(TEST_DEP)
  222. .SUFFIXES:
  223. .SECONDARY:
  224. $(BUILDDIR)lfs: $(OBJ)
  225. $(CC) $(CFLAGS) $^ $(LFLAGS) -o $@
  226. $(BUILDDIR)lfs.a: $(OBJ)
  227. $(AR) rcs $@ $^
  228. $(BUILDDIR)lfs.code.csv: $(OBJ)
  229. ./scripts/code.py $^ -q $(CODEFLAGS) -o $@
  230. $(BUILDDIR)lfs.data.csv: $(OBJ)
  231. ./scripts/data.py $^ -q $(CODEFLAGS) -o $@
  232. $(BUILDDIR)lfs.stack.csv: $(CI)
  233. ./scripts/stack.py $^ -q $(CODEFLAGS) -o $@
  234. $(BUILDDIR)lfs.struct.csv: $(OBJ)
  235. ./scripts/struct_.py $^ -q $(CODEFLAGS) -o $@
  236. $(BUILDDIR)lfs.coverage.csv: $(GCDA)
  237. ./scripts/coverage.py $^ $(patsubst %,-F%,$(SRC)) -q $(COVERAGEFLAGS) -o $@
  238. $(BUILDDIR)lfs.perf.csv: $(BENCH_PERF)
  239. ./scripts/perf.py $^ $(patsubst %,-F%,$(SRC)) -q $(PERFFLAGS) -o $@
  240. $(BUILDDIR)lfs.csv: \
  241. $(BUILDDIR)lfs.code.csv \
  242. $(BUILDDIR)lfs.data.csv \
  243. $(BUILDDIR)lfs.stack.csv \
  244. $(BUILDDIR)lfs.struct.csv
  245. ./scripts/summary.py $^ -q $(SUMMARYFLAGS) -o $@
  246. $(BUILDDIR)runners/test_runner: $(TEST_OBJ)
  247. $(CC) $(CFLAGS) $^ $(LFLAGS) -o $@
  248. $(BUILDDIR)runners/bench_runner: $(BENCH_OBJ)
  249. $(CC) $(CFLAGS) $^ $(LFLAGS) -o $@
  250. # our main build rule generates .o, .d, and .ci files, the latter
  251. # used for stack analysis
  252. $(BUILDDIR)%.o $(BUILDDIR)%.ci: %.c
  253. $(CC) -c -MMD -fcallgraph-info=su $(CFLAGS) $< -o $(BUILDDIR)$*.o
  254. $(BUILDDIR)%.s: %.c
  255. $(CC) -S $(CFLAGS) $< -o $@
  256. $(BUILDDIR)%.a.c: %.c
  257. ./scripts/prettyasserts.py -p LFS_ASSERT $< -o $@
  258. $(BUILDDIR)%.a.c: $(BUILDDIR)%.c
  259. ./scripts/prettyasserts.py -p LFS_ASSERT $< -o $@
  260. $(BUILDDIR)%.t.c: %.toml
  261. ./scripts/test.py -c $< $(TESTCFLAGS) -o $@
  262. $(BUILDDIR)%.t.c: %.c $(TESTS)
  263. ./scripts/test.py -c $(TESTS) -s $< $(TESTCFLAGS) -o $@
  264. $(BUILDDIR)%.b.c: %.toml
  265. ./scripts/bench.py -c $< $(BENCHCFLAGS) -o $@
  266. $(BUILDDIR)%.b.c: %.c $(BENCHES)
  267. ./scripts/bench.py -c $(BENCHES) -s $< $(BENCHCFLAGS) -o $@
  268. # clean everything
  269. .PHONY: clean
  270. clean:
  271. rm -f $(BUILDDIR)lfs
  272. rm -f $(BUILDDIR)lfs.a
  273. $(strip rm -f \
  274. $(BUILDDIR)lfs.csv \
  275. $(BUILDDIR)lfs.code.csv \
  276. $(BUILDDIR)lfs.data.csv \
  277. $(BUILDDIR)lfs.stack.csv \
  278. $(BUILDDIR)lfs.struct.csv \
  279. $(BUILDDIR)lfs.coverage.csv \
  280. $(BUILDDIR)lfs.perf.csv)
  281. rm -f $(OBJ)
  282. rm -f $(DEP)
  283. rm -f $(ASM)
  284. rm -f $(CI)
  285. rm -f $(TEST_RUNNER)
  286. rm -f $(TEST_TC)
  287. rm -f $(TEST_TAC)
  288. rm -f $(TEST_OBJ)
  289. rm -f $(TEST_DEP)
  290. rm -f $(TEST_CI)
  291. rm -f $(TEST_GCNO)
  292. rm -f $(TEST_GCDA)
  293. rm -f $(TEST_PERF)
  294. rm -f $(BENCH_RUNNER)
  295. rm -f $(BENCH_BC)
  296. rm -f $(BENCH_BAC)
  297. rm -f $(BENCH_OBJ)
  298. rm -f $(BENCH_DEP)
  299. rm -f $(BENCH_CI)
  300. rm -f $(BENCH_GCNO)
  301. rm -f $(BENCH_GCDA)
  302. rm -f $(BENCH_PERF)