#This program is free software: you can redistribute it and/or modify #it under the terms of the GNU General Public License as published by #the Free Software Foundation, either version 3 of the License, or #(at your option) any later version. #This program is distributed in the hope that it will be useful, #but WITHOUT ANY WARRANTY; without even the implied warranty of #MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the #GNU General Public License for more details. #You should have received a copy of the GNU General Public License #along with this program. If not, see . #Copyright 2018-2021 rexy712 #Makefile to generate a single executable from all sources in SOURCE_DIRS that end in EXT ifeq ($(OS),Windows_NT) WINDOWS::=1 endif SOURCE_DIRS::=src SOURCES::= OBJDIR::=obj DEPDIR::=$(OBJDIR)/dep LIBDIRS::=lib INCLUDE_DIRS::=include CFLAGS::=-std=c18 -Wall -pedantic -Wextra CXXFLAGS::=-std=c++17 -Wall -pedantic -Wextra DEBUG_CFLAGS::=$(CFLAGS) -O0 -g3 -ggdb DEBUG_CXXFLAGS::=$(CXXFLAGS) -O0 -g3 -ggdb RELEASE_CFLAGS::=$(CFLAGS) -O2 -Wno-strict-aliasing RELEASE_CXXFLAGS::=$(CXXFLAGS) -O2 -Wno-strict-aliasing EXT::=cpp LANG::=$(EXT) MAIN_EXECUTABLE::=tester PRE_TARGETS::= POST_TARGETS::= CLEAN_TARGETS::= RELEASE?=0 MEMCHK?=0 UNDEFCHK?=0 SAVEFLAGS?=1 ifneq ($(WINDOWS),1) #*nix settings CC::=gcc CXX::=g++ LDLIBS::= LDFLAGS::= DEBUG_LDLIBS::= DEBUG_LDFLAGS::= STRIP::=strip RANLIB::=ranlib AR::=ar AS::=as else #windows #windows settings #windows is a fuckwit MINGW_PREFIX::=x86_64-w64-mingw32- CC::=$(MINGW_PREFIX)gcc CXX::=$(MINGW_PREFIX)g++ LDLIBS::= LDFLAGS::= DEBUG_LDLIBS::= DEBUG_LDFLAGS::= STRIP::=$(MINGW_PREFIX)strip RANLIB::=$(MINGW_PREFIX)ranlib AR::=$(MINGW_PREFIX)ar AS::=$(MINGW_PREFIX)as endif #windows #Put your custom targets for PRE_TARGETS, POST_TARGETS, and CLEAN_TARGETS here: ########################################################################################################### #Everything past this point is internal BS, probably best not to touch it unless you know what you're doing #set the all target as the default target, otherwise the topmost target will run .DEFAULT_GOAL::=all #set the main target to match the output of mingw ifeq ($(WINDOWS),1) MAIN_EXECUTABLE::=$(MAIN_EXECUTABLE).exe endif #system dependant bullshit ifeq ($(OS),Windows_NT) #windows' cmd commands mkdir=mkdir $(subst /,\,$(1)) > NUL 2>&1 rm=del /F $(1) > NUL 2>&1 rmdir=rd /S /Q $(1) > NUL 2>&1 move=move /Y $(subst /,\,$(1)) $(subst /,\,$(2)) > NUL 2>&1 copy=copy /Y /B $(subst /,\,$(1)) $(subst /,\,$(2)) > NUL 2>&1 else #*nix terminal commands mkdir=mkdir -p $(1) rm=rm -f $(1) rmdir=rm -rf $(1) move=mv $(1) $(2) copy=cp $(1) $(2) endif #setup compiler and flags based on language ifeq ($(LANG),cpp) ifneq ($(RELEASE),1) COMPILER_FLAGS::=$(DEBUG_CXXFLAGS) else COMPILER_FLAGS::=$(RELEASE_CXXFLAGS) endif COMPILER::=$(CXX) else ifeq ($(LANG),c) ifneq ($(RELEASE),1) COMPILER_FLAGS::=$(DEBUG_CFLAGS) else COMPILER_FLAGS::=$(RELEASE_CFLAGS) endif COMPILER::=$(CC) endif ifneq ($(RELEASE),1) ifeq ($(MEMCHK),1) #use asan to check memory leaks/invalid accesses LDFLAGS+=-fsanitize=address -fno-omit-frame-pointer -fno-optimize-sibling-calls COMPILER_FLAGS+=-fsanitize=address -fno-omit-frame-pointer -fno-optimize-sibling-calls endif ifeq ($(UNDEFCHK),1) LDFLAGS+=-fsanitize=undefined COMPILER_FLAGS+=-fsanitize=undefined endif endif ifeq ($(SAVEFLAGS),1) CFLAGS_TMPFILE::=.cflags.tmp OLD_COMPILEFLAGS::=$(file <$(CFLAGS_TMPFILE)) LDFLAGS_TMPFILE::=.ldflags.tmp OLD_LINKFLAGS::=$(file <$(LDFLAGS_TMPFILE)) endif #add dependency tracking and include directories INTERNAL_COMPILERFLAGS=-c $(foreach dir,$(INCLUDE_DIRS),-I"$(dir)") -MMD -MP -MF"$(DEPDIR)/$(notdir $(patsubst %.o,%.d,$@))" INTERNAL_LINKFLAGS=$(foreach dir,$(LIBDIRS),-L"$(dir)") THIS_MAKEFILE_NAME::=$(lastword $(MAKEFILE_LIST)) INTERNAL_SOURCES::=$(SOURCES) $(foreach source,$(SOURCE_DIRS),$(foreach ext,$(EXT),$(wildcard $(source)/*.$(ext)))) OBJECTS::=$(addprefix $(OBJDIR)/,$(subst \,.,$(subst /,.,$(addsuffix .o,$(INTERNAL_SOURCES))))) ALL_COMPILEFLAGS=$(COMPILER_FLAGS) $(INTERNAL_COMPILERFLAGS) ifeq ($(RELEASE),1) ALL_LINKFLAGS=$(INTERNAL_LINKFLAGS) $(LDFLAGS) ALL_LDLIBS=$(LDLIBS) ALL_LINK_FLAGS_LIBS=$(ALL_LINKFLAGS) $(ALL_LDLIBS) else ALL_LINKFLAGS=$(INTERNAL_LINKFLAGS) $(LDFLAGS) $(DEBUG_LDFLAGS) ALL_LDLIBS=$(LDLIBS) $(DEBUG_LDLIBS) ALL_LINK_FLAGS_LIBS=$(ALL_LINKFLAGS) $(ALL_LDLIBS) endif #Arguments to make submake use this makefile without "Entering directory" stuff SUBMAKE_ARGUMENTS::=--no-print-directory -e -f "$(THIS_MAKEFILE_NAME)" #just a variable for a newline define \n endef .PHONY: cflags-update cflags-update: ifeq ($(SAVEFLAGS),1) ifneq ($(subst -MF"$(DEPDIR)/",-MF"$(DEPDIR)/cflags-update",$(ALL_COMPILEFLAGS)),$(OLD_COMPILEFLAGS)) $(file >$(CFLAGS_TMPFILE),$(ALL_COMPILEFLAGS)) endif endif .PHONY: ldflags-update ldflags-update: ifeq ($(SAVEFLAGS),1) ifneq ($(ALL_LINKFLAGS),$(OLD_LINK_FLAGS_LIBS)) $(file >$(LDFLAGS_TMPFILE),$(ALL_LINK_FLAGS_LIBS)) endif endif .PHONY: flags-update flags-update: cflags-update ldflags-update #default target: run targets in PRE_TARGETS, then the main executable, then POST_TARGETS .PHONY: all all: flags-update $(foreach target,$(PRE_TARGETS),@$(MAKE) $(SUBMAKE_ARGUMENTS) "$(target)"$(\n)) @$(MAKE) $(SUBMAKE_ARGUMENTS) "$(MAIN_EXECUTABLE)" $(foreach target,$(POST_TARGETS),@$(MAKE) $(SUBMAKE_ARGUMENTS) "$(target)"$(\n)) #Link executable $(MAIN_EXECUTABLE): $(OBJECTS) $(LDFLAGS_TMPFILE) $(COMPILER) $(OBJECTS) -o "$(basename $@)" $(ALL_LINKFLAGS) $(ALL_LDLIBS) ifeq ($(RELEASE),1) $(STRIP) --strip-all "$@" endif #Object target recipe define GENERATE_OBJECTS $$(OBJDIR)/$(subst \,.,$(subst /,.,$(1))).%.o: $(1)/% $(CFLAGS_TMPFILE) $$(COMPILER) $$(ALL_COMPILEFLAGS) "$$<" -o "$$@" endef define GENERATE_INDIVIDUAL_OBJECTS $$(OBJDIR)/$(subst \,.,$(subst /,.,$(1))).o: $(1) $(CFLAGS_TMPFILE) $$(COMPILER) $$(ALL_COMPILEFLAGS) "$$<" -o "$$@" endef #Create targets for object files $(foreach dir,$(SOURCE_DIRS),$(eval $(call GENERATE_OBJECTS,$(dir)))) $(foreach src,$(SOURCES),$(eval $(call GENERATE_INDIVIDUAL_OBJECTS,$(src)))) $(OBJECTS): | $(OBJDIR) $(DEPDIR) #Output directory creation $(OBJDIR): $(call mkdir,"$@") $(DEPDIR): $(call mkdir,"$@") .PHONY: clean clean: $(foreach target,$(CLEAN_TARGETS),@$(MAKE) $(SUBMAKE_ARGUMENTS) "$(target)"$(\n)) $(call rmdir,"$(DEPDIR)") $(call rmdir,"$(OBJDIR)") $(call rm,"$(MAIN_EXECUTABLE)") $(call rm,"$(MAIN_EXECUTABLE).exe") ifeq ($(SAVEFLAGS),1) $(call rm,"$(CFLAGS_TMPFILE)") $(call rm,"$(LDFLAGS_TMPFILE)") endif #header file dep tracking -include $(wildcard $(DEPDIR)/*.d)