# Makefile for sjasmplus created by Tygrys' hands.
# install/uninstall features added, CFLAGS and LDFLAGS modification by z00m's hands. [05.05.2016]
# overall optimization and beautification by mborik's hands. [05.05.2016]
# overall rewrite and extensions by Ped7g [2019-03-21 and following years]

## Some examples of my usage of this Makefile:
# make test 					- to run the CI test+example script runner
# make memcheck TEST=misc DEBUG=1		- to use valgrind on assembling sub-directory "misc" in tests
# make PREFIX=~/.local install			- to install release version into ~/.local/bin/
# make clean && make CC=gcc-14 CXX=g++-14	- to compile binary with gcc-14
# make DEBUG=1 LUA_COVERAGE=1 coverage		- to produce build/debug/coverage/* files by running the tests
# make COVERALLS_SERVICE=1 DEBUG=1 coverage	- to produce coverage data and upload them to https://coveralls.io/
# make CFLAGS_EXTRA='-m32' LDFLAGS='-ldl -m32'  - to builds 32b linux executable
# make KEEP_SYMBOLS=1 CC=clang-20 CXX=clang++-20 CFLAGS_EXTRA='-fsanitize=address' LDFLAGS='-ldl -fsanitize=address' - ASAN build
# make KEEP_SYMBOLS=1 CC=clang-20 CXX=clang++-20 CFLAGS_EXTRA='-fsanitize=undefined' LDFLAGS='-ldl -fsanitize=undefined' - UBSAN build
# to cross-compile windows exe try to use Makefile.win instead, this Makefile is now too much linux/posix only
# make callgrind TEST=integration KEEP_SYMBOLS=1 CFLAGS_EXTRA=-g && kcachegrind callgrind.out/cgo.* - callgrind profiling

EXE_BASE_NAME=sjasmplus

# Set version, dir and src.tar.xz filename
VERSION?=1.23.1
SRCTARFILE?=$(EXE_BASE_NAME)-$(VERSION)-src.tar.xz
SRCTARDIR?=$(EXE_BASE_NAME)-$(VERSION)

# Use LUA (1 = system-wide or bundled, depending on USE_BUNDLED_LUA, 0 = sjasmplus without Lua scripting)
USE_LUA?=1

# Use bundled LUA (1 = bundled source in sjasmplus repo, 0 = OS provided .so library)
USE_BUNDLED_LUA?=1

# Where to stage files when building a package
STAGEDIR?=

# Where to install resulting files (use "~/.local" to install into user's home dir)
PREFIX?=/usr/local

# define DEBUG=1 for debug build
DEBUG?=
ifdef DEBUG
KEEP_SYMBOLS?=1
endif

# strip final executable (0 = do stripping, 1 = preserve debug symbols)
KEEP_SYMBOLS?=0

# set up CC+CXX explicitly, because windows MinGW/MSYS environment don't have it set up
CC?=cc
CXX?=c++
BASH?=/usr/bin/env bash
STRIP?=strip
INSTALL?=install -c
UNINSTALL?=rm -vf
REMOVEDIR?=rm -vdf
DOCBOOKGEN?=xsltproc
MEMCHECK?=valgrind --enable-debuginfod=no --leak-check=yes --show-error-list=no --quiet
	# --leak-check=full --show-leak-kinds=all
CALLGRINDDIR?=callgrind.out
CALLGRIND?=valgrind --tool=callgrind --callgrind-out-file=$(abspath $(CALLGRINDDIR))/cgo.%p --dump-instr=yes --collect-jumps=yes --quiet
	# --simulate-cache=yes

# set up srctar
MKSRCTARDIR?=mkdir -p ../$(SRCTARDIR)
MKSRCTAR?=tar cvfJ $(SRCTARFILE) --exclude .git --exclude .github --exclude .cache --exclude build --exclude *.xz --exclude *.exe --exclude sjasmplus.res --exclude *.o --exclude LuaBridge/Tests --exclude LuaBridge/third_party ../$(SRCTARDIR)
COPYSRC?=cp -r ./* ../$(SRCTARDIR)/
RMSRCTARDIR?=rm -rf ../$(SRCTARDIR)

# all internal file names (sources, module subdirs, build dirs, ...) must be WITHOUT space!
# (i.e. every relative path from project-dir must be space-less ...)
# the project-dir itself can contain space, or any path leading up to it

BUILD_DIR=build
EXAMPLES_BUILD_DIR:=$(BUILD_DIR)/examples
TESTS_BUILD_DIR:=$(BUILD_DIR)/tests

LUA_VER?=5.5
LUA_LIBNAME?=lua$(LUA_VER)

SUBDIR_BASE=sjasm
SUBDIR_LUA=lua$(LUA_VER)
SUBDIR_LUABRIDGE=LuaBridge/Source
SUBDIR_CRC32C=crc32c
SUBDIR_DOCS=docs
SUBDIR_COV=coverage

INCDIR_LUA?=/usr/include/lua$(LUA_VER)

ifeq ($(USE_LUA), 1)
LDFLAGS+=-ldl
ifeq ($(USE_BUNDLED_LUA), 0)
_LUA_CPPFLAGS=-I$(INCDIR_LUA) -DUSE_LUA_SO_LIB
LDFLAGS+=-l$(LUA_LIBNAME)
else
_LUA_CPPFLAGS=-I$(SUBDIR_LUA)
endif
CPPFLAGS+=-DUSE_LUA -DLUA_USE_LINUX $(_LUA_CPPFLAGS) -I$(SUBDIR_LUABRIDGE)
endif

# TODO too many lua5.4 warnings: -pedantic removed
CPPFLAGS+=-Wall -DMAX_PATH=PATH_MAX -I$(SUBDIR_CRC32C)

ADD_SJ_CFLAGS=$(CFLAGS_EXTRA)

ifdef DEBUG
BUILD_DIR:=$(BUILD_DIR)/debug
ADD_SJ_CFLAGS+=-g -O0
else
BUILD_DIR:=$(BUILD_DIR)/release
ADD_SJ_CFLAGS+=-DNDEBUG -O2
# -flto		# add for LTO (link time optimization) - does make linking lot slower and result is only marginally faster
endif

# C/C++ flags (the CPPFLAGS are for preprocessor BTW, if you always wonder, like me...)
CFLAGS+=$(ADD_SJ_CFLAGS)
CXXFLAGS+=-std=c++17 $(ADD_SJ_CFLAGS)

# full path to executable
BUILD_EXE=$(BUILD_DIR)/$(EXE_BASE_NAME)

# UnitTest++ related values (slightly modified defaults)
# Unit Test exe (checks for "--unittest" and runs unit tests then)
EXE_UT_BASE_NAME=sjasm+ut
BUILD_DIR_UT=$(BUILD_DIR)+ut
SUBDIR_UT=unittest-cpp
SUBDIR_TESTS=cpp-src-tests
BUILD_EXE_UT=$(BUILD_DIR_UT)/$(EXE_UT_BASE_NAME)
LCOV_OUTPUT_FILE=$(BUILD_DIR_UT)/coverage.info

EXE_FP="$(abspath $(BUILD_EXE))"
EXE_UT_FP="$(abspath $(BUILD_EXE_UT))"

# turns list of %.c/%.cpp files into $BUILD_DIR/%.o list
define object_files
	$(addprefix $(BUILD_DIR)/, $(patsubst %.c,%.o, $(patsubst %.cpp,%.o, $(1))))
endef
define object_files_ut
	$(addprefix $(BUILD_DIR_UT)/, $(patsubst %.c,%.o, $(patsubst %.cpp,%.o, $(1))))
endef

# sjasmplus files
SRCS:=$(sort $(wildcard $(SUBDIR_BASE)/*.c) $(wildcard $(SUBDIR_BASE)/*.cpp))
OBJS:=$(call object_files,$(SRCS))
OBJS_UT:=$(call object_files_ut,$(SRCS))

ifeq ($(USE_BUNDLED_LUA), 1)
# liblua files
LUASRCS:=$(sort $(wildcard $(SUBDIR_LUA)/*.c))
LUAOBJS:=$(call object_files,$(LUASRCS))
LUAOBJS_UT:=$(call object_files_ut,$(LUASRCS))
endif

# crc32c files
CRC32CSRCS:=$(sort $(wildcard $(SUBDIR_CRC32C)/*.cpp))
CRC32COBJS:=$(call object_files,$(CRC32CSRCS))
CRC32COBJS_UT:=$(call object_files_ut,$(CRC32CSRCS))

# UnitTest++ files
UTPPSRCS:=$(sort $(wildcard $(SUBDIR_UT)/UnitTest++/*.cpp) $(wildcard $(SUBDIR_UT)/UnitTest++/Posix/*.cpp))
UTPPOBJS:=$(call object_files,$(UTPPSRCS))
TESTSSRCS:=$(sort $(wildcard $(SUBDIR_TESTS)/*.cpp))
TESTSOBJS:=$(call object_files_ut,$(TESTSSRCS))

ALL_OBJS:=$(OBJS) $(CRC32COBJS)
ALL_OBJS_UT=$(OBJS_UT) $(CRC32COBJS_UT) $(UTPPOBJS) $(TESTSOBJS)
ifeq ($(USE_LUA), 1)
ifeq ($(USE_BUNDLED_LUA), 1)
ALL_OBJS+=$(LUAOBJS)
ALL_OBJS_UT+=$(LUAOBJS_UT)
endif
endif
ALL_COVERAGE_RAW:=$(patsubst %.o,%.gcno,$(ALL_OBJS_UT)) $(patsubst %.o,%.gcda,$(ALL_OBJS_UT)) $(LCOV_OUTPUT_FILE)

# GCOV options to generate coverage files
GCOV_OPT=-rlpmab

#implicit rules to compile C/CPP files into $(BUILD_DIR)
$(BUILD_DIR)/$(SUBDIR_LUA)/%.o : $(SUBDIR_LUA)/%.c	# build Lua as C++ library, not C
	@mkdir -p $(@D)
	$(COMPILE.cc) -x c++ $(OUTPUT_OPTION) $<

$(BUILD_DIR)/%.o : %.c					# build other .c files as C (if any ever...)
	@mkdir -p $(@D)
	$(COMPILE.c) $(OUTPUT_OPTION) $<

$(BUILD_DIR)/%.o : %.cpp
	@mkdir -p $(@D)
	$(COMPILE.cc) $(OUTPUT_OPTION) $<

#implicit rules to compile C/CPP files into $(BUILD_DIR_UT) (with unit tests enabled)
$(BUILD_DIR_UT)/$(SUBDIR_LUA)/%.o : $(SUBDIR_LUA)/%.c	# build Lua as C++ library, not C
	@mkdir -p $(@D)
	$(COMPILE.cc) -x c++ -DADD_UNIT_TESTS -I$(SUBDIR_UT) $(OUTPUT_OPTION) $<

$(BUILD_DIR_UT)/%.o : %.c				# build other .c files as C (if any ever...)
	@mkdir -p $(@D)
	$(COMPILE.c) -DADD_UNIT_TESTS -I$(SUBDIR_UT) $(OUTPUT_OPTION) $<

$(BUILD_DIR_UT)/%.o : %.cpp
	@mkdir -p $(@D)
	$(COMPILE.cc) -DADD_UNIT_TESTS -I$(SUBDIR_UT) $(OUTPUT_OPTION) $<

.PHONY: all install uninstall clean docs tests test memcheck callgrind coverage coverage-lcov upx srctar

# "all" will also copy the produced binary into project root directory (to mimick old makefile)
all: $(BUILD_EXE)
	$(INSTALL) $(BUILD_EXE) $(EXE_BASE_NAME)

upx: $(BUILD_EXE)
	cp $(BUILD_EXE) $(EXE_BASE_NAME)
	upx --best $(EXE_BASE_NAME)
	EXE="$(CURDIR)/$(EXE_BASE_NAME)" $(BASH) ContinuousIntegration/test_folder_tests.sh

# make all sjasm/*.o depend on all sjasm/*.h files (no subtle dependencies, all by all affected)
$(OBJS): $(sort $(wildcard $(SUBDIR_BASE)/*.h))

$(BUILD_EXE): $(ALL_OBJS)
	$(CXX) -o $(BUILD_EXE) $(CXXFLAGS) $(ALL_OBJS) $(LDFLAGS)
ifeq ($(KEEP_SYMBOLS), 0)
	$(STRIP) $(BUILD_EXE)
endif

$(BUILD_EXE_UT): $(ALL_OBJS_UT)
	$(CXX) -o $(BUILD_EXE_UT) $(CXXFLAGS) $(ALL_OBJS_UT) $(LDFLAGS)
ifeq ($(KEEP_SYMBOLS), 0)
	$(STRIP) $(BUILD_EXE_UT)
endif

install: $(BUILD_EXE)
	$(INSTALL) -d "$(STAGEDIR)/$(PREFIX)/bin"
	$(INSTALL) $(BUILD_EXE) "$(STAGEDIR)/$(PREFIX)/bin"

uninstall:
	$(UNINSTALL) "$(STAGEDIR)/$(PREFIX)/bin/$(EXE_BASE_NAME)"

test: $(BUILD_EXE_UT)
ifdef TEST
	EXE=$(EXE_UT_FP) $(BASH) ContinuousIntegration/test_folder_tests.sh "$(TEST)"
else
	$(BUILD_EXE_UT) --unittest
	EXE=$(EXE_UT_FP) $(BASH) ContinuousIntegration/test_folder_tests.sh
	EXE=$(EXE_UT_FP) $(BASH) ContinuousIntegration/test_folder_examples.sh
endif

tests: test
	@echo "Makefile target 'tests' is deprecated, use 'test'"

memcheck: $(BUILD_EXE)
ifdef TEST
	MEMCHECK="$(MEMCHECK)" EXE=$(EXE_FP) $(BASH) ContinuousIntegration/test_folder_tests.sh "$(TEST)"
else
	MEMCHECK="$(MEMCHECK)" EXE=$(EXE_FP) $(BASH) ContinuousIntegration/test_folder_tests.sh
	MEMCHECK="$(MEMCHECK)" EXE=$(EXE_FP) $(BASH) ContinuousIntegration/test_folder_examples.sh
endif

callgrind: $(BUILD_EXE)
	@mkdir -p $(CALLGRINDDIR)
ifdef TEST
	MEMCHECK="$(CALLGRIND)" EXE=$(EXE_FP) $(BASH) ContinuousIntegration/test_folder_tests.sh "$(TEST)"
else
	MEMCHECK="$(CALLGRIND)" EXE=$(EXE_FP) $(BASH) ContinuousIntegration/test_folder_tests.sh
	MEMCHECK="$(CALLGRIND)" EXE=$(EXE_FP) $(BASH) ContinuousIntegration/test_folder_examples.sh
endif

coverage:
ifdef COVERALLS_SERVICE
	$(error gcov coverage is not meant to be used for coveralls.io, use coverage-lcov target instead)
endif
	# -fkeep-inline-functions -fkeep-static-functions seem like good idea, but they kill current coverage of header files, so nope
	$(MAKE) CFLAGS_EXTRA='--coverage' tests
	gcov $(GCOV_OPT) --object-directory $(BUILD_DIR_UT)/$(SUBDIR_BASE) $(SRCS)
	gcov $(GCOV_OPT) --object-directory $(BUILD_DIR_UT)/$(SUBDIR_CRC32C) $(CRC32CSRCS)
ifdef LUA_COVERAGE
# by default the "external" lua sources are excluded from coverage report, sjasmplus is not focusing to cover+fix lua itself
# to get full coverage report, including the lua sources, use `make DEBUG=1 LUA_COVERAGE=1 coverage`
	gcov $(GCOV_OPT) --object-directory $(BUILD_DIR_UT)/$(SUBDIR_LUA) $(LUASRCS)
endif
	@mkdir -p $(BUILD_DIR_UT)/$(SUBDIR_COV)
	mv *#*.gcov $(BUILD_DIR_UT)/$(SUBDIR_COV)/

coverage-lcov:
	$(MAKE) CFLAGS_EXTRA='--coverage' tests
	lcov --capture --base-directory ./ --directory $(BUILD_DIR_UT) --output-file $(LCOV_OUTPUT_FILE) --no-external --exclude 'unittest-cpp/*' --exclude 'LuaBridge/*' --exclude 'lua5.5/*' --exclude 'cpp-src-tests/*'
ifdef COVERALLS_SERVICE
	cp $(LCOV_OUTPUT_FILE) ./
else
	genhtml --demangle-cpp -o $(BUILD_DIR_UT)/$(SUBDIR_COV) $(LCOV_OUTPUT_FILE)
endif

docs: $(SUBDIR_DOCS)/documentation.html ;

$(SUBDIR_DOCS)/documentation.html: Makefile $(sort $(wildcard $(SUBDIR_DOCS)/*.xml) $(wildcard $(SUBDIR_DOCS)/*.xsl))
	$(DOCBOOKGEN) \
		--stringparam html.stylesheet docbook.css \
		--stringparam generate.toc "book toc" \
		-o $(SUBDIR_DOCS)/documentation.html \
		$(SUBDIR_DOCS)/docbook-xsl-ns-html-customization-linux.xsl \
		$(SUBDIR_DOCS)/documentation.xml

clean:
# delete generated example [sna, tap, com] files and leftover files from test runner run
	$(UNINSTALL) $(EXAMPLES_BUILD_DIR)/*
	$(UNINSTALL) $(TESTS_BUILD_DIR)/*
	$(UNINSTALL) \
		$(EXE_BASE_NAME) \
		$(SRCTARFILE) \
		$(BUILD_EXE) \
		$(BUILD_EXE_UT) \
		$(ALL_OBJS) \
		$(ALL_COVERAGE_RAW) \
		$(ALL_OBJS_UT) \
		$(BUILD_DIR_UT)/$(SUBDIR_COV)/**/*
	$(REMOVEDIR) \
		$(EXAMPLES_BUILD_DIR) \
		$(TESTS_BUILD_DIR) \
		$(BUILD_DIR)/$(SUBDIR_BASE) \
		$(BUILD_DIR)/$(SUBDIR_CRC32C) \
		$(BUILD_DIR)/$(SUBDIR_LUA) \
		$(BUILD_DIR)/$(SUBDIR_UT)/UnitTest++/Posix \
		$(BUILD_DIR)/$(SUBDIR_UT)/UnitTest++ \
		$(BUILD_DIR)/$(SUBDIR_UT) \
		$(BUILD_DIR) \
		$(BUILD_DIR_UT)/$(SUBDIR_BASE) \
		$(BUILD_DIR_UT)/$(SUBDIR_CRC32C) \
		$(BUILD_DIR_UT)/$(SUBDIR_LUA) \
		$(BUILD_DIR_UT)/$(SUBDIR_TESTS) \
		$(BUILD_DIR_UT)/$(SUBDIR_COV)/* \
		$(BUILD_DIR_UT)/$(SUBDIR_COV) \
		$(BUILD_DIR_UT)

srctar: clean
	$(MKSRCTARDIR)
	$(COPYSRC)
	$(MKSRCTAR)
	$(RMSRCTARDIR)
