DEV: Makefile changes (add docker support)

This commit is contained in:
Christos Choutouridis 2020-09-28 20:33:23 +03:00
parent 6588e83a68
commit d076b4ae38

View File

@ -17,12 +17,53 @@
# along with this program. If not, see <http://www.gnu.org/licenses/>. # along with this program. If not, see <http://www.gnu.org/licenses/>.
# #
# ========== Project settings ========== # ult notes:
# ==============
# This makefile provides support for unix-like local builds and
# docker based builds for gcc and clang.
# ** msvc is not currently supported.
#
# Use cases:
# 1) build localy using **local rules** in terminal.
# example:
# make -j4 MK_ARG="-std=c++14 -fconcepts" build-gcc
#
# 2) build localy using **docker based rules** via terminal or script
# in order to compile for all compilers/versions/dialects.
# example in bash:
# #!/bin/bash
# for im in gcc:8 gcc:9 gcc:10; do
# for dial in -std=c++14 -std=c++17; do
# make IMAGE="$im" MK_ARG="$dial" dock-gcc
# done
# done
#
# 3) build inside a docker instance using **local rules** in order to
# build and test using a CD/CI system.
# example in a yml file:
# image: gcc:8
# build:
# stage: build
# - make MK_ARG="-std=c++14" build-gcc
# - make MK_ARG="-std=c++14 -fconcepts" build-gcc
# - make MK_ARG="-std=c++17" build-gcc
# - make MK_ARG="-std=c++17 -fconcepts" build-gcc
# - make MK_ARG="-std=c++2a" build-gcc
# ... etc
# test:
# stage: test
# - bin/utlTest
# ... etc
# ============== Project settings ==============
# Excecutable's name # Excecutable's name
TARGET := utlTest TARGET := utlTest
# Source directories list(space seperated). Full or relative path # Source directories list(space seperated).
# Relative path, under current directory only
SRC_DIR_LIST := tests gtest SRC_DIR_LIST := tests gtest
# Include directories list(space seperated). Full or relative path # Include directories list(space seperated).
# Relative path
INC_DIR_LIST := ../include gtest INC_DIR_LIST := ../include gtest
# Exclude files list(space seperated). Filenames only. # Exclude files list(space seperated). Filenames only.
# EXC_FILE_LIST := bad.cpp old.cpp # EXC_FILE_LIST := bad.cpp old.cpp
@ -32,7 +73,7 @@ BUILD_DIR := bin
OBJ_DIR := $(BUILD_DIR)/obj OBJ_DIR := $(BUILD_DIR)/obj
DEP_DIR := $(BUILD_DIR)/.dep DEP_DIR := $(BUILD_DIR)/.dep
# ========== Compiler settings ========== # ============== Compiler settings ==============
CLANGXX := clang++ CLANGXX := clang++
GCCXX := g++ GCCXX := g++
CSIZE := size CSIZE := size
@ -42,17 +83,28 @@ REL_CFLAGS := -Wall -Wextra -O2
# Pre-defines # Pre-defines
# PRE_DEFS := MYCAB=1729 SUPER_MODE # PRE_DEFS := MYCAB=1729 SUPER_MODE
# ========== Linker settings ========== # ============== Linker settings ==============
# Linker flags # Linker flags
LDFLAGS := -pthread LDFLAGS := -pthread
# Map output file # Map output file
MAP_FILE := output.map MAP_FILE := output.map
MAP_FLAG := -Xlinker -Map=$(BUILD_DIR)/$(MAP_FILE) MAP_FLAG := -Xlinker -Map=$(BUILD_DIR)/$(MAP_FILE)
# ========== Default settings ========== # ============== Docker settings ==============
# compiler and compiler flagfs # We need:
# 1) Bind the entire project directory(the dir that icludes all the code) as volume.
# 2) In docker instance change to working directory(where the makefile is).
# For utl we use the directories `${PWD%/*}` and `test`.
# We double-$$ for double evaluating.
DOCKER_VOL_DIR := "$${PWD%/*}"
DOCKER_WRK_DIR := test
DOCKER_RUN := docker run --rm -v $(DOCKER_VOL_DIR):/usr/src/$(TARGET) -w /usr/src/$(TARGET)/$(DOCKER_WRK_DIR)
# ============== Default settings ==============
# compiler and compiler flags. By default docker is not used.
CFLAGS := $(DEB_CFLAGS) CFLAGS := $(DEB_CFLAGS)
CXX := $(GCCXX) CXX := $(GCCXX)
DOCKER :=
# #
# =========== Main body and Patterns =========== # =========== Main body and Patterns ===========
@ -67,10 +119,11 @@ EXC := $(foreach fil,$(EXC_FILE_LIST), \
SRC := $(wildcard *.cpp) SRC := $(wildcard *.cpp)
SRC += $(foreach dir,$(SRC_DIR_LIST),$(wildcard $(dir)/*.cpp)) SRC += $(foreach dir,$(SRC_DIR_LIST),$(wildcard $(dir)/*.cpp))
SRC += $(foreach dir,$(SRC_DIR_LIST),$(wildcard $(dir)/**/*.cpp)) SRC += $(foreach dir,$(SRC_DIR_LIST),$(wildcard $(dir)/**/*.cpp))
SRC := $(abspath $(filter-out $(EXC),$(SRC))) #SRC := $(abspath $(filter-out $(EXC),$(SRC)))
SRC := $(filter-out $(EXC),$(SRC))
OBJ := $(foreach file,$(SRC:%.cpp=%.o),$(OBJ_DIR)$(file)) OBJ := $(foreach file,$(SRC:%.cpp=%.o),$(OBJ_DIR)/$(file))
DEP := $(foreach file,$(SRC:%.cpp=%.d),$(DEP_DIR)$(file)) DEP := $(foreach file,$(SRC:%.cpp=%.d),$(DEP_DIR)/$(file))
# Make Dependencies pattern. # Make Dependencies pattern.
@ -80,14 +133,14 @@ DEP := $(foreach file,$(SRC:%.cpp=%.d),$(DEP_DIR)$(file))
# It is based on Tom Tromey's method. # It is based on Tom Tromey's method.
# #
# Invoke cpp to create makefile rules with dependencies for each source file # Invoke cpp to create makefile rules with dependencies for each source file
$(DEP_DIR)%.d: %.cpp $(DEP_DIR)/%.d: %.cpp
@mkdir -p $(@D) @mkdir -p $(@D)
@$(CXX) -E $(CFLAGS) $(INC) $(DEF) -MM -MT $(OBJ_DIR)$(<:.cpp=.o) -MF $@ $< @$(DOCKER) $(CXX) -E $(CFLAGS) $(INC) $(DEF) -MM -MT $(OBJ_DIR)/$(<:.cpp=.o) -MF $@ $<
# objects depent on .cpp AND dependency files, which have an empty recipe # objects depent on .cpp AND dependency files, which have an empty recipe
$(OBJ_DIR)%.o: %.cpp $(DEP_DIR)%.d $(OBJ_DIR)/%.o: %.cpp $(DEP_DIR)/%.d
@mkdir -p $(@D) @mkdir -p $(@D)
$(CXX) -c $(CFLAGS) $(INC) $(DEF) -o $@ $< $(DOCKER) $(CXX) -c $(CFLAGS) $(INC) $(DEF) -o $@ $<
# empty recipe for dependency files. This prevents make errors # empty recipe for dependency files. This prevents make errors
$(DEP): $(DEP):
@ -100,8 +153,8 @@ include $(wildcard $(DEP))
$(BUILD_DIR)/$(TARGET): $(OBJ) $(BUILD_DIR)/$(TARGET): $(OBJ)
@mkdir -p $(@D) @mkdir -p $(@D)
@echo Linking to target: $(TARGET) @echo Linking to target: $(TARGET)
@echo $(CXX) $(LDFLAGS) $(MAP_FLAG) -o $(@D)/$(TARGET) '$$(OBJ)' @echo $(DOCKER) $(CXX) $(LDFLAGS) $(MAP_FLAG) -o $(@D)/$(TARGET) '$$(OBJ)'
@$(CXX) $(LDFLAGS) $(MAP_FLAG) -o $(@D)/$(TARGET) $(OBJ) @$(DOCKER) $(CXX) $(LDFLAGS) $(MAP_FLAG) -o $(@D)/$(TARGET) $(OBJ)
@echo @echo
@echo Print size information @echo Print size information
@$(CSIZE) $(@D)/$(TARGET) @$(CSIZE) $(@D)/$(TARGET)
@ -116,43 +169,19 @@ clean:
# #
# ============ User Rules ============= # ================ Local build rules =================
# examples:
# make MK_ARG="-std=c++14 -fconcepts" build-gcc
# make MK_ARG="-std=c++17" build-clang
# #
.PHONY: build-gcc
build-gcc: CFLAGS += $(MK_ARG)
build-gcc: $(BUILD_DIR)/$(TARGET)
.PHONY: gcc14 .PHONY: build-clang
gcc14: CFLAGS += -std=c++14 build-clang: CXX := $(CLANGXX)
gcc14: $(BUILD_DIR)/$(TARGET) build-clang: CFLAGS += $(MK_ARG)
build-clang: $(BUILD_DIR)/$(TARGET)
.PHONY: gcc14_conc
gcc14_conc: CFLAGS += -std=c++14 -fconcepts
gcc14_conc: $(BUILD_DIR)/$(TARGET)
.PHONY: gcc17
gcc17: CFLAGS += -std=c++17
gcc17: $(BUILD_DIR)/$(TARGET)
.PHONY: gcc17_conc
gcc17_conc: CFLAGS += -std=c++17 -fconcepts
gcc17_conc: $(BUILD_DIR)/$(TARGET)
.PHONY: gcc2a
gcc2a: CFLAGS += -std=c++2a
gcc2a: $(BUILD_DIR)/$(TARGET)
.PHONY: clang14
clang14: CXX := $(CLANGXX)
clang14: CFLAGS += -std=c++14
clang14: $(BUILD_DIR)/$(TARGET)
.PHONY: clang17
clang17: CXX := $(CLANGXX)
clang17: CFLAGS += -std=c++17
clang17: $(BUILD_DIR)/$(TARGET)
.PHONY: clang2a
clang2a: CXX := $(CLANGXX)
clang2a: CFLAGS += -std=c++2a
clang2a: $(BUILD_DIR)/$(TARGET)
.PHONY: debug .PHONY: debug
debug: $(BUILD_DIR)/$(TARGET) debug: $(BUILD_DIR)/$(TARGET)
@ -163,3 +192,22 @@ release: clean $(BUILD_DIR)/$(TARGET)
.PHONY: all .PHONY: all
all: clean release all: clean release
#
# ================ Docker based rules ================
# examples:
# make IMAGE="gcc:8.3" MK_ARG="-std=c++14 -fconcepts" dock-gcc
# make IMAGE="a-clang-image" MK_ARG="-std=c++17" dock-clang
#
.PHONY: dock-gcc
dock-gcc: DOCKER := $(DOCKER_RUN) $(IMAGE)
dock-gcc: CFLAGS += $(MK_ARG)
dock-gcc: $(BUILD_DIR)/$(TARGET)
.PHONY: dock-clang
dock-clang: CXX := $(CLANGXX)
dock-clang: DOCKER := $(DOCKER_RUN) $(IMAGE)
dock-clang: CFLAGS += $(MK_ARG)
dock-clang: $(BUILD_DIR)/$(TARGET)