# Rules for NuttX binaries compilation of C, C++ and assembler # sources using GNU toolchain. # Interface to other rules: # kernel_HEADERS .. list of the kernel-space public header files # kernel_MODULES .. list of the kernel side modules/applications # Input variables: # NUTTX_EXPORT - export directory of NuttX build # it should contain subdirectories: # arch build include libs registry startup) # LIB_DIR - directory where compiled libraries are stored # OBJS_DIR - directory where intermediate files (.o, .map, ...) are stored # INCLUDE_DIR - where includes can be found # from config.omk or Makefile.omk # CROSS_COMPILE - target architecture tools prefix # TARGET_ARCH, DEBUG, OPTIMIZE, DEFS - forms CFLAGS # from base: SOURCES_DIR # from Makefile.omk: lib_LOADLIBES # xxx_PROGBUILTIN - list of builtin programs linked into final system image # when "all" is used then all programs provided by build are linked # xxx_PROGBUILTIN_EXCLUDE - list of builtin user or NuttX provided programs which are excluded # xxx_KMODBUILTIN - list of builtin programs linked into final system image # Output variables: # SOURCES - all the source files that needs to be compiled (except for shared library sources) # SOLIB_SOURCES - all the source files that needs to be compiled for a shared library # OBJ_EXT - extension of object files # LIB_EXT - extension of library files # LIB_PREF - prefix for library files # ASM_EXT - extension of assembler sources # Templates: # COMPILER_DEFS_template - definitions that should be defined before # the following templates can be used. The input variables needs to be # defined before evaluating this template # COMPILE_c_o_template, COMPILE_cc_o_template, COMPILE_S_o_template - # templates that create rules for compilation of sources # PROGRAM_template, LIBRARY_template, SOLIB_template - templates that # create rules for compilation of a program, library and shared # library. The rules can use rules produced by COMPILE_xxx_template. EMPTY := LD_OPTION_SEP = $(EMPTY) $(EMPTY) TARGET_STDSTARTFILES = y define COMPILER_DEFS_template OBJ_EXT ?= $$(OBJEXT) ifeq ($$(OBJ_EXT),) OBJ_EXT = .o endif LIB_EXT ?= $$(LIBEXT) ifeq ($$(LIB_EXT),) LIB_EXT = .a endif LIB_PREF = lib ASM_EXT = .S CROSS_COMPILE ?= $$(CROSSDEV) CC ?= $$(CROSS_COMPILE)gcc CXX ?= $$(CROSS_COMPILE)g++ LD ?= $$(CROSS_COMPILE)ld ifneq ($$(filter -Wl%, $$(NXFLATLDFLAGS1)$$(NXFLATLDFLAGS2)$$(LDFLAGS)),) CC_LD_OPTION = -Wl, LD_OPTION_SEP := , LD_BY_CC := 1 LD_RAW ?= $$(CROSS_COMPILE)ld endif LD_RAW ?= $$(LD) AR ?= $$(CROSS_COMPILE)ar OBJCOPY ?= $$(CROSS_COMPILE)objcopy NM ?= $$(CROSS_COMPILE)nm ifeq ($$(LDSTARTGROUP),) LDSTARTGROUP = $$(CC_LD_OPTION)--start-group LDENDGROUP = $$(CC_LD_OPTION)--end-group endif NUTTXLIBES += $$(LDSTARTGROUP) $$(LDLIBS) $$(EXTRA_LIBS) $$(LDENDGROUP) NUTTXREGISTRY += $$(NUTTX_EXPORT)/registry TARGET_ARCH += $$(ARCHCFLAGS) $$(ARCHCPUFLAGS) OPTIMIZE ?= $$(ARCHOPTIMIZATION) CFLAGS += $$(ARCHWARNINGS) CXXFLAGS += $$(ARCHCXXFLAGS) CXXFLAGS += $$(ARCHWARNINGSXX) DEFAULT_INCLUDES = -isystem $$(NUTTX_EXPORT)/include INCLUDES += CFLAGS += $$(TARGET_ARCH) $$(DEBUG) $$(OPTIMIZE) CFLAGS += -Wall CFLAGS += -I$$(SOURCES_DIR) CFLAGS += -I$$(INCLUDE_DIR) LDFLAGS := $$(filter-out -Map=%, $$(LDFLAGS)) LDFLAGS += -L"$(NUTTX_EXPORT)/libs" ifeq ($(word 2,$(AR)),) ARFLAGS = rcs else ARFLAGS = endif ifeq ($$(LDNAME),) LDNAME := $$(LDSCRIPT) endif LD_SCRIPT ?= $$(LDNAME:%=$$(NUTTX_EXPORT)/scripts/%) ifneq ($$(filter-out clean distclean, $$(MAKECMDGOALS)),) ifeq ($(LDNAME),) $$(error no LDNAME or LDSCRIPT is defined) endif ifeq ($$(wildcard $$(LD_SCRIPT)),) $$(error LD_SCRIPT='$$(LD_SCRIPT)' does not exists, correct config.target) endif endif ELF_FILE_LDSCRIPT?=$$(wildcard $$(NUTTX_EXPORT)/scripts/gnu-elf.ld) LOADLIBES += -L$$(LIB_DIR) LOADLIBES += $$(lib_LOADLIBES:%=-l%) -include $$(OBJS_DIR)/*.d SOURCES2OBJS = $$(OBJ_EXT)/.c $$(OBJ_EXT)/.cc $$(OBJ_EXT)/.cxx $$(OBJ_EXT)/.S $$(OBJ_EXT)/.s $$(OBJ_EXT)/$$(OBJ_EXT) SOURCES2OBJSLO = .lo/.c .lo/.cc .lo/.cxx .lo/.S .lo/.s .lo/.lo #%.lo: %.c # $(CC) -o $@ $(LCFLAGS) -c $< c_o_COMPILE = $$(CC) $$(DEFS) $$(DEFAULT_INCLUDES) $$(INCLUDES) $$(OMK_CPPFLAGS) \ $$(CPPFLAGS) $$(OMK_CFLAGS) $$(CFLAGS) cc_o_COMPILE = $$(CXX) $$(DEFS) $$(DEFAULT_INCLUDES) $$(INCLUDES) $$(OMK_CPPFLAGS) \ $$(CPPFLAGS) $$(CXXFLAGS) $$(OMK_CFLAGS) $$(CFLAGS) S_o_COMPILE = $$(CC) $(DEFS) $(DEFAULT_INCLUDES) $$(INCLUDES) \ $$(CPPFLAGS) $$(OMK_CFLAGS) $$(CFLAGS) $$(ASFLAGS) # Check GCC version for user build ifndef CC_MAJOR_VERSION CC_MAJOR_VERSION = $$(shell $$(CC) -dumpversion | $(SED4OMK) -e 's/\([^.]\)\..*/\1/') endif # Prepare suitable define for dependency building ifeq ($$(CC_MAJOR_VERSION),2) CC_DEPFLAGS = -Wp,-MD,"$$@.d.tmp" else CC_DEPFLAGS = -MT $$@ -MD -MP -MF "$$@.d.tmp" endif endef # COMPILER_DEFS_template # Syntax: $(call COMPILE_c_o_template,,,) define COMPILE_c_o_template $(2): $(1) $$(GEN_HEADERS) @$(QUIET_CMD_ECHO) " CC $$@" $(Q) if $$(c_o_COMPILE) $$(CC_DEPFLAGS) $(3) -o $$@ -c $$< ; \ then mv -f "$$@.d.tmp" "$$@.d" ; \ else rm -f "$$@.d.tmp" ; exit 1; \ fi endef # Syntax: $(call COMPILE_cc_o_template,,,) define COMPILE_cc_o_template $(2): $(1) $$(GEN_HEADERS) @$(QUIET_CMD_ECHO) " CXX $$@" $(Q) if $$(cc_o_COMPILE) $$(CC_DEPFLAGS) $(3) -o $$@ -c $$< ; \ then mv -f "$$@.d.tmp" "$$@.d" ; \ else rm -f "$$@.d.tmp" ; exit 1; \ fi endef # Syntax: $(call COMPILE_S_o_template,,,) define COMPILE_S_o_template $(2): $(1) $$(GEN_HEADERS) @$(QUIET_CMD_ECHO) " AS $$@" $(Q) if $$(S_o_COMPILE) -D__ASSEMBLY__ $$(CC_DEPFLAGS) $(3) -o $$@ -c $$< ; \ then if [ -e "$$@.d.tmp" ] ; then mv -f "$$@.d.tmp" "$$@.d" ; fi ; \ else rm -f "$$@.d.tmp" ; exit 1; \ fi endef # Syntax: $(call PROGRAM_template,,,) define PROGRAM_template GEN_SOURCES += $$($(1)_GEN_SOURCES) $(foreach x, $(SOURCES2OBJS), $(1)_OBJS += $$(patsubst %$(notdir $(x)),%$(dir $(x)),$$(filter %$(notdir $(x)),\ $$($(1)_SOURCES) $$($(1)_GEN_SOURCES))) ) $(1)_OBJS += $$($(1)_EMBEDROMFS:%=%_img$$(OBJ_EXT)) $(1)_OBJS := $$(sort $$($(1)_OBJS:%/=%)) SOURCES += $$($(1)_SOURCES) EMBEDROMFS += $$($(1)_EMBEDROMFS) ifneq ($$($(1)_PROGBUILTIN),) ifneq ($$($(1)_PROGBUILTIN),all) $(1)_PROGBUILTIN_TMP = $$($(1)_PROGBUILTIN) else $(1)_PROGBUILTIN_TMP1 = $$(wildcard $$(USER_REGISTRY_DIR)/*.pbi) $(1)_PROGBUILTIN_TMP = $$($(1)_PROGBUILTIN_TMP1:$$(USER_REGISTRY_DIR)/%.pbi=%) endif $(1)_PROGBUILTIN_LIST = $$(filter-out $$($(1)_PROGBUILTIN_EXCLUDE),$$($(1)_PROGBUILTIN_TMP)) $(1)_PROGBUILTIN_OTHER=$$(filter-out $(1),$$($(1)_PROGBUILTIN_LIST)) $(1)_PROGBUILTIN_PBI=$$($(1)_PROGBUILTIN_OTHER:%=$$(USER_REGISTRY_DIR)/%.pbi) $(1)_PROGBUILTIN_LDAT=$$($(1)_PROGBUILTIN_LIST:%=$$(USER_REGISTRY_DIR)/%.ldat) $(1)_PROGBUILTIN_PDAT=$$($(1)_PROGBUILTIN_LIST:%=$$(USER_REGISTRY_DIR)/%.pdat) \ $$(filter-out $$($(1)_PROGBUILTIN_EXCLUDE:%=$(NUTTXREGISTRY)/%.pdat),\ $$(wildcard $(NUTTXREGISTRY)/*.pdat)) $(1)_PROGBUILTIN_BDAT=$$($(1)_PROGBUILTIN_LIST:%=$$(USER_REGISTRY_DIR)/%.bdat) \ $$(filter-out $$($(1)_PROGBUILTIN_EXCLUDE:%=$(NUTTXREGISTRY)/%.bdat),\ $$(wildcard $(NUTTXREGISTRY)/*.bdat)) ifneq ($$($(1)_KMODBUILTIN),all) $(1)_KMODBUILTIN_LIBS = $$($(1)_KMODBUILTIN:%=$$(KERN_MODULES_DIR)/lib%.ka) else $(1)_KMODBUILTIN_LIBS = $$(wildcard $$(KERN_MODULES_DIR)/lib*.ka) endif ifneq ($$($(1)_PROGBUILTIN_PDAT),) $(1)_BUILTINTABLE = $(OBJS_DIR)/$(1)-builtintab.c $$($(1)_BUILTINTABLE): $$($(1)_PROGBUILTIN_PDAT) $$($(1)_PROGBUILTIN_BDAT) $(Q) cat $$($(1)_PROGBUILTIN_PDAT) >$$@ $(Q) echo "#include " >>$$@ $(Q) echo "#include " >>$$@ $(Q) echo "const struct builtin_s g_builtins[] = {" >>$$@ $(Q) cat $$($(1)_PROGBUILTIN_BDAT) >>$$@ $(Q) echo "{ NULL, 0, 0, 0 }};" >>$$@ $(Q) echo "const int g_builtin_count = sizeof(g_builtins) / sizeof(g_builtins[0]);" >>$$@ GEN_SOURCES += $$($(1)_BUILTINTABLE) $(1)_BUILTINWRAPMAIN = $(OBJS_DIR)/$(1)-builtinwrapmain.c $$($(1)_BUILTINWRAPMAIN): $(Q) echo "#include " >$$@ $(Q) echo "#include " >>$$@ $(Q) echo "int main(int argc, FAR char *argv[]);" >>$$@ $(Q) echo "int weak_function $(1)_main(int argc, FAR char *argv[]) {return main(argc, argv);}" >>$$@ GEN_SOURCES += $$($(1)_BUILTINWRAPMAIN) endif endif ifneq ($$(LD_SCRIPT$(3:%=-%)),) $(1)$(3:%=-%)_LD_SCRIPT_FN = $$(LD_SCRIPT$(3:%=-%)) else $(1)$(3:%=-%)_LD_SCRIPT_FN = $$(LD_SCRIPT) endif ifeq ($$(dir $$($(1)$(3:%=-%)_LD_SCRIPT_FN)),) $(1)$(3:%=-%)_LD_SCRIPT = $$(LIB_DIR)/$$($(1)$(3:%=-%)_LD_SCRIPT_FN) else $(1)$(3:%=-%)_LD_SCRIPT = $$($(1)$(3:%=-%)_LD_SCRIPT_FN) endif $(1)$(3:%=-%)_LDFLAGS = $$($(1)$(3:%=-%)_LD_SCRIPT:%=$$(CC_LD_OPTION)-T$$(LD_OPTION_SEP)%) $(2)/$(1)$(3:%=-%): $$($(1)_OBJS) $$($(1)_BUILTINTABLE:%.c=%$$(OBJ_EXT)) $$($(1)_BUILTINWRAPMAIN:%.c=%$$(OBJ_EXT)) $$($(1)_PROGBUILTIN_LDAT) $$($(1)$(3:%=-%)_LD_SCRIPT) $$($(1)_KMODBUILTIN_LIBS) @$(QUIET_CMD_ECHO) " LINK $$@" $(Q) $$(if $$(LD_BY_CC) , $$(if $$(filter %.cc,$$($(1)_SOURCES)) , \ $$(CXX) $$(CPPFLAGS) $$(OMK_CPPFLAGS) $$(OMK_CXXFLAGS) $$(CXXFLAGS) , \ $$(CC) $$(CPPFLAGS) $$(OMK_CPPFLAGS) $$(OMK_CFLAGS) $$(CFLAGS) ) , \ $$(LD) \ ) \ $$(OMK_LDFLAGS) $$(LDFLAGS) $$($(1)$(3:%=-%)_LDFLAGS) $$(CC_LD_OPTION)-Map$$(LD_OPTION_SEP)$(1)$(3:%=-%).map \ $$(HEAD_OBJ:%=$$(NUTTX_EXPORT)/startup/%) \ $$($(1)_OBJS) $$($(1)_MOREOBJS) $$($(1)_BUILTINTABLE:%.c=%$$(OBJ_EXT)) $$($(1)_BUILTINWRAPMAIN:%.c=%$$(OBJ_EXT)) \ $$($(1)_PROGBUILTIN_PBI) $$(LOADLIBES) $$($(1)_LIBS:%=-l%) $$($(1)_KMODBUILTIN_LIBS) \ $$(shell for i in $$($(1)_PROGBUILTIN_LDAT); do cat $$$$i ; done) $$(NUTTXLIBES) \ -o $$@ @echo "$(2)/$(1)$(3:%=-%): \\" >$(OBJS_DIR)/$(1)$(3:%=-%).exe.d @$(SED4OMK) -n -e 's|^LOAD \(.*\)$$$$| \1 \&|p' $(OBJS_DIR)/$(1)$(3:%=-%).map|tr '&' '\134' | grep -v '^ linker [^ ]' >>$(OBJS_DIR)/$(1)$(3:%=-%).exe.d @echo >>$(OBJS_DIR)/$(1).exe.d endef # Syntax: $(call ELF_template,,) define ELF_template $(2)/$(1).elf: $$($(1)_OBJS) @$(QUIET_CMD_ECHO) " ELF $$@" $(Q) $$(LD_RAW) \ $$(filter -m%,$$(LDFLAGS:$$(CC_LD_OPTION)%=%)) \ -r -e main -T $$(ELF_FILE_LDSCRIPT) \ -Map $(1).elf.map \ $$($(1)_OBJS) $$($(1)_MOREOBJS) $$(LOADLIBES) $$($(1)_LIBS:%=-l%) \ -o $$@ @echo "$(2)/$(1).elf: \\" >$(OBJS_DIR)/$(1).elf.d @echo " $(ELF_FILE_LDSCRIPT) \\" >>$(OBJS_DIR)/$(1).elf.d @$(SED4OMK) -n -e 's|^LOAD \(.*\)$$$$| \1 \&|p' $(OBJS_DIR)/$(1).elf.map|tr '&' '\134' | grep -v '^ linker [^ ]' >>$(OBJS_DIR)/$(1).elf.d @echo >>$(OBJS_DIR)/$(1).elf.d endef # Syntax: $(call PROGBUILTIN_template,,,) define PROGBUILTIN_template $(1)_PRIORITY ?= SCHED_PRIORITY_DEFAULT $(1)_STACKSIZE ?= 2048 $(2)/$(1).pbi: $$($(1)_OBJS) @$(QUIET_CMD_ECHO) " PROGMOD $$@" $(Q) $(LD_RAW) \ $$(filter -m%,$$(LDFLAGS:$$(CC_LD_OPTION)%=%)) \ -r -x --defsym=$(1)_main=main \ -Map $(1).pbi.map \ $$($(1)_OBJS) $$($(1)_MOREOBJS) \ -o $(OBJS_DIR)/$(1).pbo $(Q) $(OBJCOPY) --keep-global-symbol=$(1)_main $(OBJS_DIR)/$(1).pbo $$@ @echo "$(2)/$(1).pbi: \\" >$(OBJS_DIR)/$(1).pbi.d @$(SED4OMK) -n -e 's|^LOAD \(.*\)$$$$| \1 \&|p' $(OBJS_DIR)/$(1).pbi.map|tr '&' '\134' | grep -v '^ linker [^ ]' >>$(OBJS_DIR)/$(1).pbi.d @echo >>$(OBJS_DIR)/$(1).pbi.d @echo "{ \"$1\", $$($(1)_PRIORITY), $$($(1)_STACKSIZE), $(1)_main }," >$(3)/$(1).bdat @echo "int $(1)_main(int argc, char *argv[]);" >$(3)/$(1).pdat @echo "$$(LOADLIBES) $$($(1)_LIBS:%=-l%)" >$(3)/$(1).ldat endef # Rules for other output formats (can be specified by OUTPUT_FORMATS) %.bin: % @$(QUIET_CMD_ECHO) " OBJCOPY $@" $(Q) $(OBJCOPY) --output-target=binary -S $< $@ %.hex: % @$(QUIET_CMD_ECHO) " OBJCOPY $@" $(Q) $(OBJCOPY) --output-target=ihex -S $< $@ %.srec: % @$(QUIET_CMD_ECHO) " OBJCOPY $@" $(Q) $(OBJCOPY) --output-target=srec -S $< $@ # Syntax: $(call LIBRARY_template,) define LIBRARY_template GEN_SOURCES += $$($(1)_GEN_SOURCES) $(foreach x, $(SOURCES2OBJS), $(1)_OBJS += $$(patsubst %$(notdir $(x)),%$(dir $(x)),$$(filter %$(notdir $(x)),\ $$($(1)_SOURCES) $$($(1)_GEN_SOURCES))) ) $(1)_OBJS += $$($(1)_EMBEDROMFS:%=%_img$$(OBJ_EXT)) $(1)_OBJS := $$(sort $$($(1)_OBJS:%/=%)) SOURCES += $$($(1)_SOURCES) EMBEDROMFS += $$($(1)_EMBEDROMFS) $(LIB_DIR)/lib$(1)$$(LIB_EXT): $$($(1)_OBJS) @$(QUIET_CMD_ECHO) " AR $$@" $(Q) $(AR) $(ARFLAGS) $$@ $$^ endef # Syntax: $(call SOLIB_template,) define SOLIB_template SOLIB_GEN_SOURCES += $$($(1)_GEN_SOURCES) $(foreach x, $(SOURCES2OBJSLO), $(1)_OBJSLO += $$(patsubst %$(notdir $(x)),%$(dir $(x)),$$(filter %$(notdir $(x)),\ $$($(1)_SOURCES) $$($(1)_GEN_SOURCES))) ) $(1)_OBJSLO := $$(sort $$($(1)_OBJSLO:%/=%)) SOLIB_OBJS += $$($(1)_OBJSLO) SOLIB_SOURCES += $$($(1)_SOURCES) $(LIB_DIR)/lib$(1).so: $$($(1)_OBJSLO) @$(QUIET_CMD_ECHO) " LINK $$@" $(Q) $(LD_RAW) \ $$(filter -m%,$$(LDFLAGS:$$(CC_LD_OPTION)%=%) \ --shared --soname=lib$(1).so -o $$@ $$^ endef ifneq ($(bin_PROGRAMS),) library-pass_HOOKS += force_builtin_library_hook .PHONY: force_builtin_library_hook force_builtin_library_hook: endif # Syntax: $(call ROMFS_template,,,) FIXME: Is this correct? define ROMFS_template .PHONY: $(2) $(2): $(Q) genromfs -f $(2).tmp -V $(1) -d $(3) $(Q) if ( [ ! -e $(2) ] || ! cmp -s $(2).tmp $(2) ) ; then cp $(2).tmp $(2) ; fi $(2)$$(OBJ_EXT): $(2) @$(QUIET_CMD_ECHO) " ROMFS $$@" $(Q) $(LD_RAW) -r --accept-unknown-input-arch -b binary \ $$(filter -m%,$$(LDFLAGS:$$(CC_LD_OPTION)%=%)) \ --defsym=$(2)=_binary_$(2)_start \ --defsym=$(2)_size=_binary_$(2)_size \ --defsym=$(2)_start=_binary_$(2)_start \ --defsym=$(2)_end=_binary_$(2)_end \ -o $$@.tmp $$^ $(Q) $(OBJCOPY) --rename-section .data=.rodata,alloc,load,readonly,data,contents \ $$@.tmp $$@ endef