1 # -*- mode: makefile;-*-
3 # Copyright (C) 1999-2020 Apple Inc. All rights reserved.
5 # MakeInc.cmd contains command paths for use during
6 # the build, as well as make fragments and text
7 # strings that may be evaluated as utility functions.
11 # Commands for the build environment
15 # Build Logging and Verbosity
26 ERR = $(ECHO) > /dev/stderr
31 PRINTF = printf > /dev/null
37 # Helper functions for logging operations.
39 LOG_PFX_LEN_ADJ = $(LOG_PFX_LEN)
40 LOG = $(PRINTF) "$2%$4s$(Color0) $3%s$(Color0)\n" "$1"
44 # Concise logging puts all logs on the same line (CSI K to clear and
46 LOG = $(PRINTF) "$2%$4s$(Color0) $3%s$(Color0)\033[K\r" "$1"
49 _LOG_COMP = $(call LOG,$1,$(ColorC),$(ColorF),$(LOG_PFX_LEN_ADJ))
50 _LOG_HOST = $(call LOG,$1,$(ColorH),$(ColorF),$(LOG_PFX_LEN))
51 _LOG_HOST_LINK = $(call LOG,$1,$(ColorH),$(ColorLF),$(LOG_PFX_LEN))
54 LOG_LDFILELIST = $(call LOG,LDFILELIST,$(ColorL),$(ColorLF),$(LOG_PFX_LEN_ADJ))
55 LOG_MIG = $(call LOG,MIG,$(ColorM),$(ColorF),$(LOG_PFX_LEN_ADJ))
56 LOG_LD = $(call LOG,LD,$(ColorL),$(ColorF),$(LOG_PFX_LEN_ADJ))
57 LOG_ALIGN = $(call LOG,--------->,$(Color0),$(Color0),$(LOG_PFX_LEN))
59 # Compiling/machine-specific operations.
60 LOG_CC = $(call _LOG_COMP,CC)
61 LOG_CXX = $(call _LOG_COMP,C++)
62 LOG_AS = $(call _LOG_COMP,AS)
63 LOG_LTO = $(call _LOG_COMP,LTO)
64 LOG_SYMBOLSET = $(call _LOG_COMP,SYMSET)
65 LOG_SYMBOLSETPLIST = $(call _LOG_COMP,SYMSETPLIST)
67 # Host-side operations.
68 LOG_IIG = $(call _LOG_HOST,IIG)
69 LOG_HOST_CC = $(call _LOG_HOST,CC)
70 LOG_HOST_LD = $(call _LOG_HOST,LD)
71 LOG_HOST_CODESIGN = $(call _LOG_HOST,CODESIGN)
72 LOG_HOST_BISON = $(call _LOG_HOST,BISON)
73 LOG_HOST_FLEX = $(call _LOG_HOST,FLEX)
74 LOG_INSTALL = $(call _LOG_HOST,INSTALL)
75 LOG_INSTALLSYM = $(call _LOG_HOST,INSTALLSYM)
76 LOG_INSTALLHDR = $(call _LOG_HOST,INSTALLHDR)
77 LOG_INSTALLMACROS = $(call _LOG_HOST,INSTALLMACROS)
78 LOG_INSTALLPY = $(call _LOG_HOST,INSTALLPY)
79 LOG_MAN = $(call _LOG_HOST,MAN)
80 LOG_MANLINK = $(call _LOG_HOST,MANLINK)
81 LOG_ALIAS = $(call _LOG_HOST,ALIAS)
82 LOG_STRIP = $(call _LOG_HOST,STRIP)
83 LOG_DSYMUTIL = $(call _LOG_HOST,DSYMUTIL)
84 LOG_LIBTOOL = $(call _LOG_HOST,LIBTOOL)
85 LOG_FILEPREP = $(call _LOG_HOST,FILEPREP)
87 # Host-side linking operations.
88 LOG_GENASSYM = $(call _LOG_HOST_LINK,GENASSYM)
89 LOG_GENERATE= $(call _LOG_HOST_LINK,GENERATE)
90 LOG_CTFCONVERT = $(call _LOG_HOST_LINK,CTFCONVERT)
91 LOG_CTFMERGE = $(call _LOG_HOST_LINK,CTFMERGE)
92 LOG_CTFINSERT = $(call _LOG_HOST_LINK,CTFINSERT)
93 LOG_DSYMUTIL = $(call _LOG_HOST_LINK,DSYMUTIL)
94 LOG_SUPPORTED_KPI = $(call _LOG_HOST_LINK,SUPPORTED_KPI)
100 XCRUN = /usr/bin/xcrun -verbose
103 _vstdout = > /dev/null
104 _vstderr = 2&> /dev/null
105 XCRUN = /usr/bin/xcrun
108 VERBOSE_GENERATED_MAKE_FRAGMENTS = NO
115 HOST_SDKROOT ?= macosx
117 # SDKROOT may be passed as a shorthand like "iphoneos.internal". We
118 # must resolve these to a full path and override SDKROOT.
120 ifeq ($(SDKROOT_RESOLVED),)
121 export SDKROOT_RESOLVED := $(shell $(XCRUN) -sdk $(SDKROOT) -show-sdk-path)
122 ifeq ($(strip $(SDKROOT)_$(SDKROOT_RESOLVED)),/_)
123 export SDKROOT_RESOLVED := /
126 override SDKROOT = $(SDKROOT_RESOLVED)
128 ifeq ($(HOST_SDKROOT_RESOLVED),)
129 export HOST_SDKROOT_RESOLVED := $(shell $(XCRUN) -sdk $(HOST_SDKROOT) -show-sdk-path)
130 ifeq ($(strip $(HOST_SDKROOT_RESOLVED)),)
131 export HOST_SDKROOT_RESOLVED := /
134 override HOST_SDKROOT = $(HOST_SDKROOT_RESOLVED)
137 export PLATFORMPATH := $(shell $(XCRUN) -sdk $(SDKROOT) -show-sdk-platform-path)
138 export PLATFORM := $(shell echo $(PLATFORMPATH) | sed 's,^.*/\([^/]*\)\.platform$$,\1,')
140 export PLATFORM := MacOSX
141 else ifeq ($(shell echo $(PLATFORM) | tr A-Z a-z),watchos)
142 export PLATFORM := WatchOS
146 ifeq ($(PLATFORM),MacOSX)
147 ifeq (DriverKit,$(shell echo $(SDKROOT_RESOLVED) | sed 's,^.*/\([^/1-9]*\)[1-9][^/]*\.sdk$$,\1,'))
148 export PLATFORM := DriverKit
149 export DRIVERKIT ?= 1
150 export DRIVERKITROOT ?= /System/DriverKit
151 export DRIVERKITRUNTIMEROOT = $(DRIVERKITROOT)/Runtime
155 ifeq ($(SDKVERSION),)
156 export SDKVERSION := $(shell $(XCRUN) -sdk $(SDKROOT) -show-sdk-version)
159 # CC/CXX get defined by make(1) by default, so we can't check them
160 # against the empty string to see if they haven't been set
161 ifeq ($(origin CC),default)
162 export CC := $(shell $(XCRUN) -sdk $(SDKROOT) -find clang)
164 ifeq ($(origin CXX),default)
165 export CXX := $(shell $(XCRUN) -sdk $(SDKROOT) -find clang++)
168 export MIG := $(shell $(XCRUN) -sdk $(SDKROOT) -find mig)
171 export MIGCOM := $(shell $(XCRUN) -sdk $(SDKROOT) -find migcom)
174 export MIGCC := $(CC)
177 export IIG := $(shell $(XCRUN) -sdk $(SDKROOT) -find iig)
180 export STRIP := $(shell $(XCRUN) -sdk $(SDKROOT) -find strip)
183 export LIPO := $(shell $(XCRUN) -sdk $(SDKROOT) -find lipo)
186 export LIBTOOL := $(shell $(XCRUN) -sdk $(SDKROOT) -find libtool)
189 export OTOOL := $(shell $(XCRUN) -sdk $(SDKROOT) -find otool)
192 export NM := $(shell $(XCRUN) -sdk $(SDKROOT) -find nm)
195 export UNIFDEF := $(shell $(XCRUN) -sdk $(SDKROOT) -find unifdef)
198 export DSYMUTIL := $(shell $(XCRUN) -sdk $(SDKROOT) -find dsymutil)
200 ifeq ($(CTFCONVERT),)
201 export CTFCONVERT := $(shell $(XCRUN) -sdk $(SDKROOT) -find ctfconvert)
204 export CTFMERGE := $(shell $(XCRUN) -sdk $(SDKROOT) -find ctfmerge)
207 export CTFINSERT := $(shell $(XCRUN) -sdk $(SDKROOT) -find ctf_insert)
210 export NMEDIT := $(shell $(XCRUN) -sdk $(SDKROOT) -find nmedit)
216 SUPPORTED_EMBEDDED_PLATFORMS := iPhoneOS iPhoneOSNano tvOS AppleTVOS WatchOS BridgeOS
217 SUPPORTED_SIMULATOR_PLATFORMS := iPhoneSimulator iPhoneNanoSimulator tvSimulator AppleTVSimulator WatchSimulator
218 SUPPORTED_PLATFORMS := MacOSX DriverKit $(SUPPORTED_SIMULATOR_PLATFORMS) $(SUPPORTED_EMBEDDED_PLATFORMS)
220 # Platform-specific tools
221 EDM_DBPATH ?= $(PLATFORMPATH)/usr/local/standalone/firmware/device_map.db
223 # Scripts or tools we build ourselves
225 # setsegname - Rename segments in a Mach-O object file
226 # kextsymboltool - Create kext pseudo-kext Mach-O kexts binaries
227 # decomment - Strip out comments to detect whether a file is comments-only
228 # installfile - Atomically copy files, esp. when multiple architectures
229 # are trying to install the same target header
230 # replacecontents - Write contents to a file and update modtime *only* if
233 SEG_HACK = $(OBJROOT)/SETUP/setsegname/setsegname
234 KEXT_CREATE_SYMBOL_SET = $(OBJROOT)/SETUP/kextsymboltool/kextsymboltool
235 DECOMMENT = $(OBJROOT)/SETUP/decomment/decomment
236 NEWVERS = $(SRCROOT)/config/newvers.pl
237 INSTALL = $(OBJROOT)/SETUP/installfile/installfile
238 REPLACECONTENTS = $(OBJROOT)/SETUP/replacecontents/replacecontents
239 JSONCOMPILATIONDB = $(OBJROOT)/SETUP/json_compilation_db/json_compilation_db
248 MKDIR = /bin/mkdir -p
251 XARGS = /usr/bin/xargs
253 BASENAME = /usr/bin/basename
254 DIRNAME = /usr/bin/dirname
256 TOUCH = /usr/bin/touch
260 PLUTIL = /usr/bin/plutil
263 # Command to generate host binaries. Intentionally not
264 # $(CC), which controls the target compiler
266 ifeq ($(HOST_OS_VERSION),)
267 export HOST_OS_VERSION := $(shell sw_vers -productVersion)
270 export HOST_CC := $(shell $(XCRUN) -sdk $(HOST_SDKROOT) -find clang)
273 export HOST_FLEX := $(shell $(XCRUN) -sdk $(HOST_SDKROOT) -find flex)
275 ifeq ($(HOST_BISON),)
276 export HOST_BISON := $(shell $(XCRUN) -sdk $(HOST_SDKROOT) -find bison)
279 export HOST_GM4 := $(shell $(XCRUN) -sdk $(HOST_SDKROOT) -find gm4)
281 ifeq ($(HOST_CODESIGN),)
282 export HOST_CODESIGN := /usr/bin/codesign
284 ifeq ($(HOST_CODESIGN_ALLOCATE),)
285 export HOST_CODESIGN_ALLOCATE := $(shell $(XCRUN) -sdk $(HOST_SDKROOT) -find codesign_allocate)
289 # The following variables are functions invoked with "call", and thus
290 # behave similarly to externally compiled commands
293 # $(1) is an expanded kernel config from a TARGET_CONFIGS_UC tuple
294 # $(2) is an expanded arch config from a TARGET_CONFIGS_UC tuple
295 # $(3) is an expanded machine config from a TARGET_CONFIGS_UC tuple
296 _function_create_build_configs_join = $(strip $(1))^$(strip $(2))^$(strip $(3))
298 # $(1) is an un-expanded kernel config from a TARGET_CONFIGS_UC tuple
299 # $(2) is an un-expanded arch config from a TARGET_CONFIGS_UC tuple
300 # $(3) is an un-expanded machine config from a TARGET_CONFIGS_UC tuple
301 _function_create_build_configs_do_expand = $(call _function_create_build_configs_join, \
302 $(if $(filter DEFAULT,$(1)), \
303 $(DEFAULT_KERNEL_CONFIG), \
306 $(if $(filter DEFAULT,$(2)), \
307 $(DEFAULT_ARCH_CONFIG), \
310 $(if $(filter DEFAULT,$(3)), \
311 $(if $(filter DEFAULT,$(2)), \
312 $(DEFAULT_$(DEFAULT_ARCH_CONFIG)_MACHINE_CONFIG), \
313 $(DEFAULT_$(strip $(2))_MACHINE_CONFIG) \
319 # $(1) is an un-expanded TARGET_CONFIGS_UC list, which must be consumed
320 # 3 elements at a time
321 function_create_build_configs = $(sort \
323 $(call _function_create_build_configs_do_expand, \
328 $(if $(word 4,$(1)), \
329 $(call function_create_build_configs, \
330 $(wordlist 4,$(words $(1)),$(1)) \
336 # Similar to build configs, but alias configs are a 4-tuple
338 # $(1) is an expanded kernel config from a TARGET_CONFIGS_ALIASES_UC tuple
339 # $(2) is an expanded arch config from a TARGET_CONFIGS_ALIASES_UC tuple
340 # $(3) is an expanded kernel machine config from a TARGET_CONFIGS_ALIASES_UC tuple
341 # $(4) is an expanded SoC platform config from a TARGET_CONFIGS_ALIASES_UC tuple,
342 # which should be an alias of $(3)
343 _function_create_alias_configs_join = $(strip $(1))^$(strip $(2))^$(strip $(3))^$(strip $(4))
345 _function_create_alias_configs_do_expand = $(call _function_create_alias_configs_join, \
346 $(if $(filter DEFAULT,$(1)), \
347 $(DEFAULT_KERNEL_CONFIG), \
350 $(if $(filter DEFAULT,$(2)), \
351 $(DEFAULT_ARCH_CONFIG), \
358 function_create_alias_configs = $(sort \
360 $(call _function_create_alias_configs_do_expand, \
366 $(if $(word 5,$(1)), \
367 $(call function_create_alias_configs, \
368 $(wordlist 5,$(words $(1)),$(1)) \
374 # $(1) is a fully-expanded kernel config
375 # $(2) is a fully-expanded arch config
376 # $(3) is a fully-expanded machine config. "NONE" is not represented in the objdir path
377 function_convert_target_config_uc_to_objdir = $(if $(filter NONE,$(3)),$(strip $(1))_$(strip $(2)),$(strip $(1))_$(strip $(2))_$(strip $(3)))
379 # $(1) is a fully-expanded build config (like "RELEASE^X86_64^NONE")
380 function_convert_build_config_to_objdir = $(call function_convert_target_config_uc_to_objdir, \
381 $(word 1,$(subst ^, ,$(1))), \
382 $(word 2,$(subst ^, ,$(1))), \
383 $(word 3,$(subst ^, ,$(1))) \
386 # $(1) is a fully-expanded build config (like "RELEASE^X86_64^NONE")
387 function_extract_kernel_config_from_build_config = $(word 1,$(subst ^, ,$(1)))
388 function_extract_arch_config_from_build_config = $(word 2,$(subst ^, ,$(1)))
389 function_extract_machine_config_from_build_config = $(word 3,$(subst ^, ,$(1)))
391 # $(1) is an input word
392 # $(2) is a list of colon-separate potential substitutions like "FOO:BAR BAZ:QUX"
393 # $(3) is a fallback if no substitutions were made
394 function_substitute_word_with_replacement = $(strip $(if $(2), \
395 $(if $(filter $(word 1,$(subst :, ,$(word 1,$(2)))),$(1)), \
396 $(word 2,$(subst :, ,$(word 1,$(2)))), \
397 $(call function_substitute_word_with_replacement,$(1),$(wordlist 2,$(words $(2)),$(2)),$(3))), \
402 # You can't assign a variable to an empty space without these
405 space := $(empty) $(empty)
408 # $(1) is the number to increment
409 NUM32 = x x x x x x x x x x x x x x x x x x x x x x x x x x x x x x x x
410 increment = $(words x $(wordlist 1,$(1),$(NUM32)))
411 decrement = $(words $(wordlist 2,$(1),$(NUM32)))
413 # Create a sequence from 1 to $(1)
414 # F(N) = if N > 0: return F(N-1) + "N" else: return ""
415 sequence = $(if $(wordlist 1,$(1),$(NUM32)),$(call sequence,$(call decrement,$(1))) $(1),)
417 # Reverse a list of words in $(1)
418 reverse = $(if $(word 2,$(1)),$(call reverse,$(wordlist 2,$(words $(1)),$(1)))) $(word 1,$(1))