Makefile 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. ifndef VERBOSE
  2. .SILENT:
  3. endif
  4. ifdef silent
  5. SILENT = $(silent)
  6. endif
  7. ifdef SILENT
  8. SUB_IS_SILENT := $(silent)
  9. endif
  10. override SILENT = false
  11. STARTING_MAKEFILE := $(firstword $(MAKEFILE_LIST))
  12. ROOT_MAKEFILE := $(lastword $(MAKEFILE_LIST))
  13. ROOT_DIR := $(dir $(ROOT_MAKEFILE))
  14. ifeq ($(ROOT_DIR),)
  15. ROOT_DIR := .
  16. endif
  17. ABS_STARTING_MAKEFILE := $(abspath $(STARTING_MAKEFILE))
  18. ABS_ROOT_MAKEFILE := $(abspath $(ROOT_MAKEFILE))
  19. ABS_STARTING_DIR := $(dir $(ABS_STARTING_MAKEFILE))
  20. ABS_ROOT_DIR := $(dir $(ABS_ROOT_MAKEFILE))
  21. STARTING_DIR := $(subst $(ABS_ROOT_DIR),,$(ABS_STARTING_DIR))
  22. PATH_ELEMENTS := $(subst /, ,$(STARTING_DIR))
  23. MAKEFILE_INCLUDED=yes
  24. define NEXT_PATH_ELEMENT
  25. $$(eval CURRENT_PATH_ELEMENT := $$(firstword $$(PATH_ELEMENTS)))
  26. $$(eval PATH_ELEMENTS := $$(wordlist 2,9999,$$(PATH_ELEMENTS)))
  27. endef
  28. $(eval $(call NEXT_PATH_ELEMENT))
  29. ifeq ($(CURRENT_PATH_ELEMENT),keyboards)
  30. $(eval $(call NEXT_PATH_ELEMENT))
  31. KEYBOARD := $(CURRENT_PATH_ELEMENT)
  32. $(eval $(call NEXT_PATH_ELEMENT))
  33. ifeq ($(CURRENT_PATH_ELEMENT),keymaps)
  34. $(eval $(call NEXT_PATH_ELEMENT))
  35. KEYMAP := $(CURRENT_PATH_ELEMENT)
  36. else ifneq ($(CURRENT_PATH_ELEMENT),)
  37. SUBPROJECT := $(CURRENT_PATH_ELEMENT)
  38. $(eval $(call NEXT_PATH_ELEMENT))
  39. ifeq ($(CURRENT_PATH_ELEMENT),keymaps)
  40. $(eval $(call NEXT_PATH_ELEMENT))
  41. KEYMAP := $(CURRENT_PATH_ELEMENT)
  42. endif
  43. endif
  44. endif
  45. # Only consider folders with makefiles, to prevent errors in case there are extra folders
  46. KEYBOARDS := $(notdir $(patsubst %/Makefile,%,$(wildcard $(ROOT_DIR)/keyboards/*/Makefile)))
  47. #Compability with the old make variables
  48. ifdef keyboard
  49. KEYBOARD := $(keyboard)
  50. endif
  51. ifdef sub
  52. SUBPROJECT := $(sub)
  53. endif
  54. ifdef subproject
  55. SUBPROJECT := $(subproject)
  56. endif
  57. ifdef keymap
  58. KEYMAP := $(keymap)
  59. endif
  60. #$(info Keyboard: $(KEYBOARD))
  61. #$(info Keymap: $(KEYMAP))
  62. #$(info Subproject: $(SUBPROJECT))
  63. #$(info Keyboards: $(KEYBOARDS))
  64. .DEFAULT_GOAL := all
  65. ifneq ($(KEYMAP),)
  66. ifeq ($(SUBPROJECT),)
  67. .DEFAULT_GOAL := $(KEYBOARD)-$(KEYMAP)
  68. else
  69. .DEFAULT_GOAL := $(KEYBOARD)-$(SUBPROJECT)-$(KEYMAP)
  70. endif
  71. else ifneq ($(SUBPROJECT),)
  72. .DEFAULT_GOAL := $(KEYBOARD)-$(SUBPROJECT)-allkm
  73. else ifneq ($(KEYBOARD),)
  74. .DEFAULT_GOAL := $(KEYBOARD)-allsp-allkm
  75. endif
  76. # Compare the start of the RULE_VARIABLE with the first argument($1)
  77. # If the rules equals $1 or starts with $1-, RULE_FOUND is set to true
  78. # and $1 is removed from the RULE variable
  79. # Otherwise the RULE_FOUND variable is set to false
  80. # The function is a bit tricky, since there's no built in $(startswith) function
  81. define COMPARE_AND_REMOVE_FROM_RULE_HELPER
  82. ifeq ($1,$$(RULE))
  83. RULE:=
  84. RULE_FOUND := true
  85. else
  86. STARTDASH_REMOVED=$$(subst START$1-,,START$$(RULE))
  87. ifneq ($$(STARTDASH_REMOVED),START$$(RULE))
  88. RULE_FOUND := true
  89. RULE := $$(STARTDASH_REMOVED)
  90. else
  91. RULE_FOUND := false
  92. endif
  93. endif
  94. endef
  95. COMPARE_AND_REMOVE_FROM_RULE = $(eval $(call COMPARE_AND_REMOVE_FROM_RULE_HELPER,$1))$(RULE_FOUND)
  96. # Recursively try to find a match
  97. # $1 The list to be checked
  98. # If a match is found, then RULE_FOUND is set to true
  99. # and MATCHED_ITEM to the item that was matched
  100. define TRY_TO_MATCH_RULE_FROM_LIST_HELPER
  101. ifneq ($1,)
  102. ifeq ($$(call COMPARE_AND_REMOVE_FROM_RULE,$$(firstword $1)),true)
  103. MATCHED_ITEM := $$(firstword $1)
  104. else
  105. $$(eval $$(call TRY_TO_MATCH_RULE_FROM_LIST_HELPER,$$(wordlist 2,9999,$1)))
  106. endif
  107. endif
  108. endef
  109. TRY_TO_MATCH_RULE_FROM_LIST = $(eval $(call TRY_TO_MATCH_RULE_FROM_LIST_HELPER,$1))$(RULE_FOUND)
  110. define ALL_IN_LIST_LOOP
  111. OLD_RULE$1 := $$(RULE)
  112. $$(eval $$(call $1,$$(ITEM$1)))
  113. RULE := $$(OLD_RULE$1)
  114. endef
  115. define PARSE_ALL_IN_LIST
  116. $$(foreach ITEM$1,$2,$$(eval $$(call ALL_IN_LIST_LOOP,$1)))
  117. endef
  118. define PARSE_RULE
  119. RULE := $1
  120. COMMANDS :=
  121. ifeq ($$(call COMPARE_AND_REMOVE_FROM_RULE,allkb),true)
  122. $$(eval $$(call PARSE_ALL_KEYBOARDS))
  123. else ifeq ($$(call TRY_TO_MATCH_RULE_FROM_LIST,$$(KEYBOARDS)),true)
  124. $$(eval $$(call PARSE_KEYBOARD,$$(MATCHED_ITEM)))
  125. else ifneq ($$(KEYBOARD),)
  126. $$(eval $$(call PARSE_KEYBOARD,$$(KEYBOARD)))
  127. else
  128. $$(info make: *** No rule to make target '$1'. Stop.)
  129. exit 1
  130. endif
  131. endef
  132. # $1 = Keyboard
  133. define PARSE_KEYBOARD
  134. CURRENT_KB := $1
  135. # A subproject is any keyboard subfolder with a makefile
  136. SUBPROJECTS := $$(notdir $$(patsubst %/Makefile,%,$$(wildcard $(ROOT_DIR)/keyboards/$$(CURRENT_KB)/*/Makefile)))
  137. ifeq ($$(call COMPARE_AND_REMOVE_FROM_RULE,allsp),true)
  138. $$(eval $$(call PARSE_ALL_SUBPROJECTS))
  139. else ifeq ($$(call COMPARE_AND_REMOVE_FROM_RULE,defaultsp),true)
  140. $$(eval $$(call PARSE_SUBPROJECT,defaultsp))
  141. else ifeq ($$(call TRY_TO_MATCH_RULE_FROM_LIST,$$(SUBPROJECTS)),true)
  142. $$(eval $$(call PARSE_SUBPROJECT,$$(MATCHED_ITEM)))
  143. else ifneq ($$(SUBPROJECT),)
  144. $$(eval $$(call PARSE_SUBPROJECT,$$(SUBPROJECT)))
  145. else
  146. # If there's no matching subproject, we assume it's the default
  147. # This will allow you to leave the subproject part of the target out
  148. $$(eval $$(call PARSE_SUBPROJECT,defaultsp))
  149. endif
  150. endef
  151. define PARSE_ALL_KEYBOARDS
  152. $$(eval $$(call PARSE_ALL_IN_LIST,PARSE_KEYBOARD,$(KEYBOARDS)))
  153. endef
  154. # $1 Subproject
  155. define PARSE_SUBPROJECT
  156. ifeq ($1,defaultsp)
  157. SUBPROJECT_DEFAULT=
  158. $$(eval include $(ROOT_DIR)/keyboards/$$(CURRENT_KB)/Makefile)
  159. CURRENT_SP := $$(SUBPROJECT_DEFAULT)
  160. else
  161. CURRENT_SP := $1
  162. endif
  163. # If current subproject is empty (the default was not defined), and we have a list of subproject
  164. # then make all
  165. ifeq ($$(CURRENT_SP),)
  166. ifneq ($$(SUBPROJECTS),)
  167. CURRENT_SP := allsp
  168. endif
  169. endif
  170. ifneq ($$(CURRENT_SP),allsp)
  171. KEYMAPS := $$(notdir $$(patsubst %/.,%,$$(wildcard $(ROOT_DIR)/keyboards/$$(CURRENT_KB)/keymaps/*/.)))
  172. ifneq ($$(CURRENT_SP),)
  173. SP_KEYMAPS := $$(notdir $$(patsubst %/.,%,$$(wildcard $(ROOT_DIR)/keyboards/$$(CURRENT_KB)/$$(CURRENT_SP)/keymaps/*/.)))
  174. KEYMAPS := $$(sort $$(KEYMAPS) $$(SP_KEYMAPS))
  175. endif
  176. ifeq ($$(RULE),)
  177. $$(eval $$(call PARSE_ALL_KEYMAPS))
  178. else ifeq ($$(call COMPARE_AND_REMOVE_FROM_RULE,allkm),true)
  179. $$(eval $$(call PARSE_ALL_KEYMAPS))
  180. else ifeq ($$(call TRY_TO_MATCH_RULE_FROM_LIST,$$(KEYMAPS)),true)
  181. $$(eval $$(call PARSE_KEYMAP,$$(MATCHED_ITEM)))
  182. else ifneq ($$(KEYMAP),)
  183. $$(eval $$(call PARSE_KEYMAP,$$(KEYMAP)))
  184. else
  185. ifeq ($$(CURRENT_SP),)
  186. $$(info make: *** No rule to make target '$$(CURRENT_KB)-$$(RULE)'. Stop.)
  187. else
  188. $$(info make: *** No rule to make target '$$(CURRENT_KB)-$$(CURRENT_SP)-$$(RULE)'. Stop.)
  189. endif
  190. exit 1
  191. endif
  192. else
  193. $$(eval $$(call PARSE_ALL_IN_LIST,PARSE_SUBPROJECT,$(SUBPROJECTS)))
  194. endif
  195. endef
  196. define PARSE_ALL_SUBPROJECTS
  197. ifeq ($$(SUBPROJECTS),)
  198. $$(eval $$(call PARSE_SUBPROJECT,defaultsp))
  199. else
  200. $$(eval $$(call PARSE_ALL_IN_LIST,PARSE_SUBPROJECT,$$(SUBPROJECTS)))
  201. endif
  202. endef
  203. # $1 Keymap
  204. define PARSE_KEYMAP
  205. CURRENT_KM = $1
  206. COMMAND := COMMAND_KEYBOARD_$$(CURRENT_KB)_SUBPROJECT_$(CURRENT_SP)_KEYMAP_$$(CURRENT_KM)
  207. COMMANDS += $$(COMMAND)
  208. ifeq ($$(CURRENT_SP),)
  209. KB_SP := $(CURRENT_KB)
  210. else
  211. KB_SP := $(CURRENT_KB)/$$(CURRENT_SP)
  212. endif
  213. KB_SP := $(BOLD)$$(KB_SP)$(NO_COLOR)
  214. MAKE_VARS := KEYBOARD=$$(CURRENT_KB) SUBPROJECT=$$(CURRENT_SP) KEYMAP=$$(CURRENT_KM)
  215. MAKE_VARS += VERBOSE=$(VERBOSE) COLOR=$(COLOR)
  216. MAKE_CMD := $$(MAKE) -r -R -C $(ROOT_DIR) -f build_keyboard.mk
  217. MAKE_MSG := Compiling $$(KB_SP) with $(BOLD)$$(CURRENT_KM)$(NO_COLOR)
  218. MAKE_MSG_FORMAT := $(AWK) '{ printf "%-118s", $$$$0;}'
  219. COMMAND_true_$$(COMMAND) := \
  220. printf "$$(MAKE_MSG)" | \
  221. $$(MAKE_MSG_FORMAT); \
  222. LOG=$$$$($$(MAKE_CMD) $$(MAKE_VARS) SILENT=true 2>&1) ; \
  223. if [ $$$$? -gt 0 ]; \
  224. then $$(PRINT_ERROR_PLAIN); \
  225. elif [ "$$$$LOG" != "" ] ; \
  226. then $$(PRINT_WARNING_PLAIN); \
  227. else \
  228. $$(PRINT_OK); \
  229. fi;
  230. COMMAND_false_$$(COMMAND) := \
  231. printf "$$(MAKE_MSG)\n" | \
  232. $$(MAKE_MSG_FORMAT); \
  233. $$(MAKE_CMD) $$(MAKE_VARS) SILENT=false;
  234. endef
  235. define PARSE_ALL_KEYMAPS
  236. $$(eval $$(call PARSE_ALL_IN_LIST,PARSE_KEYMAP,$$(KEYMAPS)))
  237. endef
  238. define SET_SILENT_MODE
  239. ifdef SUB_IS_SILENT
  240. SILENT_MODE := $(SUB_IS_SILENT)
  241. else ifeq ($$(words $$(COMMANDS)),1)
  242. SILENT_MODE := false
  243. else
  244. SILENT_MODE := true
  245. endif
  246. endef
  247. include $(ROOT_DIR)/message.mk
  248. RUN_COMMAND = \
  249. $(COMMAND_$(SILENT_MODE)_$(COMMAND))
  250. # Allow specifying just the subproject, in the keyboard directory, which will compile all keymaps
  251. SUBPROJECTS := $(notdir $(patsubst %/Makefile,%,$(wildcard ./*/Makefile)))
  252. .PHONY: $(SUBPROJECTS)
  253. $(SUBPROJECTS): %: %-allkm
  254. .PHONY: %
  255. %:
  256. cmp --version >/dev/null 2>&1; if [ $$? -gt 0 ]; then printf "$(MSG_NO_CMP)"; exit 1; fi;
  257. git submodule status --recursive 2>/dev/null | \
  258. while IFS= read -r x; do \
  259. case "$$x" in \
  260. \ *) ;; \
  261. *) printf "$(MSG_SUBMODULE_DIRTY)";break;; \
  262. esac \
  263. done
  264. $(eval $(call PARSE_RULE,$@))
  265. $(eval $(call SET_SILENT_MODE))
  266. +$(foreach COMMAND,$(COMMANDS),$(RUN_COMMAND))
  267. .PHONY: all
  268. all: all-keyboards
  269. .PHONY: all-keyboards
  270. all-keyboards: allkb-allsp-allkm
  271. .PHONY: all-keyboards-defaults
  272. all-keyboards-defaults: allkb-allsp-default
  273. GIT_VERSION := $(shell git describe --abbrev=6 --dirty --always --tags 2>/dev/null || date +"%Y-%m-%d-%H:%M:%S")
  274. BUILD_DATE := $(shell date +"%Y-%m-%d-%H:%M:%S")
  275. $(shell echo '#define QMK_VERSION "$(GIT_VERSION)"' > $(ROOT_DIR)/quantum/version.h)
  276. $(shell echo '#define QMK_BUILDDATE "$(BUILD_DATE)"' >> $(ROOT_DIR)/quantum/version.h)