252 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			Makefile
		
	
	
	
	
	
			
		
		
	
	
			252 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			Makefile
		
	
	
	
	
	
| #
 | |
| # PDS HW2 Makefile
 | |
| #
 | |
| # Copyright (C) 2024 Christos Choutouridis <christos@choutouridis.net>
 | |
| #
 | |
| # This program is free software: you can redistribute it and/or modify
 | |
| # it under the terms of the GNU Lesser 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 Lesser General Public License for more details.
 | |
| #
 | |
| # You should have received a copy of the GNU Lesser General Public License
 | |
| # along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
| #
 | |
| 
 | |
| # ============== Project settings ==============
 | |
| # Project's name
 | |
| PROJECT         := PDS_homework_2
 | |
| 
 | |
| # Excecutable's name
 | |
| TARGET          := bitonic
 | |
| 
 | |
| # Source directories list(space seperated). Makefile-relative path, UNDER current directory.
 | |
| SRC_DIR_LIST    := src test test/gtest
 | |
| 
 | |
| # Include directories list(space seperated). Makefile-relative path.
 | |
| INC_DIR_LIST    := include \
 | |
|                    test \
 | |
|                    test/gtest/ \
 | |
|                    /usr/lib/x86_64-linux-gnu/openmpi/include/ \
 | |
|                    src
 | |
| 
 | |
| # Exclude files list(space seperated). Filenames only.
 | |
| # EXC_FILE_LIST := bad.cpp old.cpp
 | |
| 
 | |
| # Build directories
 | |
| BUILD_DIR       := bin
 | |
| OBJ_DIR         := $(BUILD_DIR)/obj
 | |
| DEP_DIR         := $(BUILD_DIR)/.dep
 | |
| 
 | |
| OUTPUT_DIR      := out
 | |
| 
 | |
| # ========== Compiler settings ==========
 | |
| # Compiler flags for debug and release
 | |
| DEB_CFLAGS      := -DDEBUG -g3 -Wall -Wextra -std=c11 -fopenmp
 | |
| REL_CFLAGS      := -Wall -Wextra -O3 -std=c11 -fopenmp
 | |
| DEB_CXXFLAGS    := -DDEBUG -g3 -Wall -Wextra -std=c++17 -fopenmp
 | |
| REL_CXXFLAGS    := -Wall -Wextra -O3 -std=c++17 -fopenmp
 | |
| 
 | |
| # Pre-defines
 | |
| # PRE_DEFS := MYCAB=1729 SUPER_MODE
 | |
| PRE_DEFS        := _GLIBCXX_PARALLEL
 | |
| 
 | |
| # ============== Linker settings ==============
 | |
| # Linker flags (example: -pthread -lm)
 | |
| LDFLAGS         := -pthread -fopenmp
 | |
| 
 | |
| # Map output file
 | |
| MAP_FILE        := output.map
 | |
| MAP_FLAG        := -Xlinker -Map=$(BUILD_DIR)/$(MAP_FILE)
 | |
| 
 | |
| # ============== Docker settings ==============
 | |
| # We need:
 | |
| #  - Bind the entire project directory(the dir that icludes all the code) as volume.
 | |
| #  - In docker instance, change to working directory(where the makefile is).
 | |
| DOCKER_VOL_DIR  := $(shell pwd)
 | |
| DOCKER_WRK_DIR  :=
 | |
| DOCKER_RUN      := docker run --rm
 | |
| DOCKER_FLAGS    := -v $(DOCKER_VOL_DIR):/usr/src/$(PROJECT) -w /usr/src/$(PROJECT)/$(DOCKER_WRK_DIR)
 | |
| 
 | |
| # docker invoke mechanism (edit with care)
 | |
| #   note:
 | |
| #   Here, `DOCKER` variable is empty. Rules can assign `DOCKER := DOCKER_CMD` when docker
 | |
| #   functionality is needed.
 | |
| DOCKER_CMD      = $(DOCKER_RUN) $(DOCKER_FLAGS) $(IMAGE)
 | |
| DOCKER          :=
 | |
| 
 | |
| # ============== Tool selection ==============
 | |
| # compiler and compiler flags.
 | |
| CSIZE           := size
 | |
| CFLAGS          := $(DEB_CFLAGS)
 | |
| CXXFLAGS        := $(DEB_CXXFLAGS)
 | |
| CXX             := g++ #mpic++
 | |
| CC              := gcc #mpicc
 | |
| 
 | |
| #
 | |
| # =========== Main body and Patterns ===========
 | |
| #
 | |
| 
 | |
| #ifeq ($(OS), Windows_NT)
 | |
| #	TARGET := $(TARGET).exe
 | |
| #endif
 | |
| INC     := $(foreach dir,$(INC_DIR_LIST),-I$(dir))
 | |
| DEF     := $(foreach def,$(PRE_DEFS),-D$(def))
 | |
| EXC     := $(foreach fil,$(EXC_FILE_LIST),                              \
 | |
|                $(foreach dir,$(SRC_DIR_LIST),$(wildcard $(dir)/$(fil))) \
 | |
|            )
 | |
| # source files. object and dependencies list
 | |
| # recursive search into current and source directories
 | |
| SRC     := $(wildcard *.cpp)
 | |
| SRC     += $(foreach dir,$(SRC_DIR_LIST),$(wildcard $(dir)/*.cpp))
 | |
| SRC     += $(foreach dir,$(SRC_DIR_LIST),$(wildcard $(dir)/**/*.cpp))
 | |
| SRC     := $(filter-out $(EXC),$(SRC))
 | |
| #SRC     := $(abspath $(SRC))
 | |
| 
 | |
| OBJ     := $(foreach file,$(SRC:%.cpp=%.o),$(OBJ_DIR)/$(file))
 | |
| DEP     := $(foreach file,$(SRC:%.cpp=%.d),$(DEP_DIR)/$(file))
 | |
| 
 | |
| 
 | |
| # Make Dependencies pattern.
 | |
| # This little trick enables recompilation only when dependencies change
 | |
| # and it does so for changes both in source AND header files ;)
 | |
| # 
 | |
| # It is based on Tom Tromey's method.
 | |
| # 
 | |
| # Invoke cpp to create makefile rules with dependencies for each source file
 | |
| $(DEP_DIR)/%.d: %.c
 | |
| 	@mkdir -p $(@D)
 | |
| 	@$(DOCKER) $(CC) -E $(CFLAGS) $(INC) $(DEF) -MM -MT $(OBJ_DIR)/$(<:.c=.o) -MF $@ $<
 | |
| 
 | |
| # c file objects depent on .c AND dependency files, which have an empty recipe 
 | |
| $(OBJ_DIR)/%.o: %.c $(DEP_DIR)/%.d
 | |
| 	@mkdir -p $(@D)
 | |
| 	@$(DOCKER) $(CC) -c $(CFLAGS) $(INC) $(DEF) -o $@ $<
 | |
| 
 | |
| $(DEP_DIR)/%.d: %.cpp
 | |
| 	@mkdir -p $(@D)
 | |
| 	@$(DOCKER) $(CXX) -E $(CXXFLAGS) $(INC) $(DEF) -MM -MT $(OBJ_DIR)/$(<:.cpp=.o) -MF $@ $<
 | |
| 
 | |
| # cpp file objects depent on .cpp AND dependency files, which have an empty recipe 
 | |
| $(OBJ_DIR)/%.o: %.cpp $(DEP_DIR)/%.d
 | |
| 	@mkdir -p $(@D)
 | |
| 	@$(DOCKER) $(CXX) -c $(CXXFLAGS) $(INC) $(DEF) -o $@ $<
 | |
| 
 | |
| # empty recipe for dependency files. This prevents make errors
 | |
| $(DEP):
 | |
| 
 | |
| # now include all dependencies
 | |
| # After all they are makefile dependency rules ;)
 | |
| include $(wildcard $(DEP))
 | |
| 
 | |
| # main target rule
 | |
| $(BUILD_DIR)/$(TARGET): $(OBJ)
 | |
| 	@mkdir -p $(@D)
 | |
| 	@echo Linking to target: $(TARGET)
 | |
| 	@echo $(DOCKER) $(CXX) '$$(OBJ)' $(LDFLAGS) $(MAP_FLAG) -o $(@D)/$(TARGET)
 | |
| 	@$(DOCKER) $(CXX) $(OBJ) $(LDFLAGS) $(MAP_FLAG) -o $(@D)/$(TARGET)
 | |
| 	@echo
 | |
| 	@echo Print size information
 | |
| 	@$(CSIZE) $(@D)/$(TARGET)
 | |
| 	@echo Done
 | |
| 
 | |
| 
 | |
| #
 | |
| # ================ Default local build rules =================
 | |
| # example:
 | |
| # make debug
 | |
| 
 | |
| .DEFAULT_GOAL := all
 | |
| 
 | |
| .PHONY: clean
 | |
| clean:
 | |
| 	@echo Cleaning build directories
 | |
| 	@rm -rf $(OBJ_DIR)
 | |
| 	@rm -rf $(DEP_DIR)
 | |
| 	@rm -rf $(BUILD_DIR)
 | |
| 
 | |
| debug: CFLAGS := $(DEB_CFLAGS)
 | |
| debug: $(BUILD_DIR)/$(TARGET)
 | |
| 
 | |
| release: CFLAGS := $(REL_CFLAGS)
 | |
| release: $(BUILD_DIR)/$(TARGET)
 | |
| 
 | |
| #
 | |
| # ================ Build rules =================
 | |
| #
 | |
| 
 | |
| # Local or inside HPC rules
 | |
| distbubbletonic: CC := mpicc
 | |
| distbubbletonic: CXX := mpic++
 | |
| distbubbletonic: CFLAGS := $(REL_CFLAGS) -DCODE_VERSION=BUBBLETONIC
 | |
| distbubbletonic: CXXFLAGS := $(REL_CXXFLAGS) -DCODE_VERSION=BUBBLETONIC
 | |
| distbubbletonic: TARGET := distbubbletonic
 | |
| distbubbletonic: $(BUILD_DIR)/$(TARGET)
 | |
| 	@mkdir -p $(OUTPUT_DIR)
 | |
| 	cp $(BUILD_DIR)/$(TARGET) $(OUTPUT_DIR)/$(TARGET)
 | |
| 
 | |
| distbitonic: CC := mpicc
 | |
| distbitonic: CXX := mpic++
 | |
| distbitonic: CFLAGS := $(REL_CFLAGS) -DCODE_VERSION=BITONIC
 | |
| distbitonic: CXXFLAGS := $(REL_CXXFLAGS) -DCODE_VERSION=BITONIC
 | |
| distbitonic: TARGET := distbitonic
 | |
| distbitonic: $(BUILD_DIR)/$(TARGET)
 | |
| 	@mkdir -p $(OUTPUT_DIR)
 | |
| 	cp $(BUILD_DIR)/$(TARGET) $(OUTPUT_DIR)/$(TARGET)
 | |
| 
 | |
| deb_distbubbletonic: CC := mpicc
 | |
| deb_distbubbletonic: CXX := mpic++
 | |
| deb_distbubbletonic: CFLAGS := $(DEB_CFLAGS) -DCODE_VERSION=BUBBLETONIC -DDEBUG
 | |
| deb_distbubbletonic: CXXFLAGS := $(DEB_CXXFLAGS) -DCODE_VERSION=BUBBLETONIC -DDEBUG
 | |
| deb_distbubbletonic: TARGET := deb_distbubbletonic
 | |
| deb_distbubbletonic: $(BUILD_DIR)/$(TARGET)
 | |
| 	@mkdir -p $(OUTPUT_DIR)
 | |
| 	cp $(BUILD_DIR)/$(TARGET) $(OUTPUT_DIR)/$(TARGET)
 | |
| 
 | |
| deb_distbitonic: CC := mpicc
 | |
| deb_distbitonic: CXX := mpic++
 | |
| deb_distbitonic: CFLAGS := $(DEB_CFLAGS) -DCODE_VERSION=BITONIC -DDEBUG
 | |
| deb_distbitonic: CXXFLAGS := $(DEB_CXXFLAGS) -DCODE_VERSION=BITONIC -DDEBUG
 | |
| deb_distbitonic: TARGET := deb_distbitonic
 | |
| deb_distbitonic: $(BUILD_DIR)/$(TARGET)
 | |
| 	@mkdir -p $(OUTPUT_DIR)
 | |
| 	cp $(BUILD_DIR)/$(TARGET) $(OUTPUT_DIR)/$(TARGET)
 | |
| 
 | |
| tests: CC := mpicc
 | |
| tests: CXX := mpic++
 | |
| tests: CFLAGS := $(DEB_CFLAGS) -DCODE_VERSION=BITONIC -DDEBUG -DTESTING
 | |
| tests: CXXFLAGS := $(DEB_CXXFLAGS) -DCODE_VERSION=BITONIC -DDEBUG -DTESTING
 | |
| tests: TARGET := tests
 | |
| tests: $(BUILD_DIR)/$(TARGET)
 | |
| 	@mkdir -p $(OUTPUT_DIR)
 | |
| 	cp $(BUILD_DIR)/$(TARGET) $(OUTPUT_DIR)/$(TARGET)
 | |
| 
 | |
| perfbitonic: CC := mpicc
 | |
| perfbitonic: CXX := mpic++
 | |
| perfbitonic: CFLAGS := $(REL_CFLAGS) -g -DCODE_VERSION=BITONIC
 | |
| perfbitonic: CXXFLAGS := $(REL_CXXFLAGS) -g -DCODE_VERSION=BITONIC
 | |
| perfbitonic: TARGET := perfbitonic
 | |
| perfbitonic: $(BUILD_DIR)/$(TARGET)
 | |
| 	@mkdir -p $(OUTPUT_DIR)
 | |
| 	cp $(BUILD_DIR)/$(TARGET) $(OUTPUT_DIR)/$(TARGET)
 | |
| 
 | |
| 
 | |
| hpc-build:
 | |
| 	make clean
 | |
| 	make distbubbletonic
 | |
| 	make clean
 | |
| 	make distbitonic
 | |
| 	make clean
 | |
| 	make tests
 | |
| 
 | |
| 
 | |
| all: debug distbubbletonic distbitonic
 | |
| # Note:
 | |
| #	Add a gcc based make rule here in order for clangd to successfully scan the project files.
 | |
| #	Otherwise we do not need the gcc build.
 | |
| 
 |