################################################################################
# This makefile builds APC, the Asset Package Compiler for Henge, on the system.
################################################################################
+# Driver sources
+DRIVERS := apc testapc
+
# Yacc
YACC := bison
YFLAGS ?= -v -d -Wall
-YCMD = $(YACC) $(YFLAGS) $<
+YCMD = $(strip $(YACC) $(YFLAGS) $(if $2,$(dir $2))$1)
+YCMD += $(if $2,&& mv $(notdir $(1:%.y=%.tab.[ch])) $(dir $2))
# Ragel
RLC ?= ragel
RLFALGS ?= -C
-RLCMD = $(RLC) $(RLFLAGS) -o $@ $<
+RLCMD = $(strip $(RLC) $(RLFLAGS) $(if $2,-o $2 $(dir $2))$1)
# C
CC ?= gcc
-CFLAGS ?= -Wall -Werror
-CCMD = $(CC) $(CFLAGS) $(CPPFLAGS) -o $@ $<
+CFLAGS ?= -Wall
+CCMD = $(strip $(CC) $(CFLAGS) $(CPPFLAGS) -c $(if $2,-o $2) $1)
# Linker
-LD ?= ld
-LDFLAGS ?=
-LDLIBS ?=
-LDCMD = $(LD) $(LDFLAGS) $(LDLIBS) -o $@ $^
+LD ?= ld
+LDFLAGS ?=
+LDLIBS ?= -lc -lunistring
+apcLIBS ?=
+apc-dLIBS ?=
+LDCMD = $(strip $(LD) $(LDFLAGS) $(if $2,-o $2) $1) $(LDLIBS) $($1LIBS)
# APC is built from Ragel, Bison and C source code only.
ySRC := $(shell find ./src -type f -name '*.y')
hGEN := $(ySRC:%.y=%.tab.h)
cGEN := $(strip $(ySRC:%.y=%.tab.c) $(rlSRC:%.rl=%.c))
-# Functions
-cGENDEP = $(if $(wildcard $1),$(subst src/,,$(filter-out $1 \ %:,$(shell $(CC) -MM -MG $1))),$(info <$1>))
-ldFLAGS = $(strip $(LDFLAGS) $(LDLIBS) $(VA_ARGS))
+# Filter all other driver objects out of each driver's link commands.
+OBJ := $(patsubst %.c,%.o,$(cSRC) $(cGEN))
+$(foreach drv,$(DRIVERS),\
+$(eval OTHERS := $(filter-out $(drv),$(DRIVERS)))\
+$(eval $(drv)SRC := $(filter-out $(OTHERS:%=\%/%.o),$(OBJ)))\
+$(eval $(drv)-dSRC := $(filter-out $(OTHERS:%=\%/%-d.o),$(OBJ:%.o=%-d.o))))
-# Rules
-apcSRC := $(patsubst %.c,%.o,$(cSRC) $(cGEN))
-apc-dSRC := $(patsubst %.o,%-d.o,$(apcSRC))
+# Unless cleaning, deps should be generated for each source file
+ifeq (,$(filter clean,$(MAKECMDGOALS)))
+cGENDEP = $(if $(wildcard $1),$(subst $(dir $1),,$(filter-out $1 \ %:,$(shell $(CC) -MM -MG $1))),\
+$(info [<$1>: no deps - file not found]))
+endif
+
+# Construct the S2S function for generating source files during prerequisite
+# expansion:
+# If we are cleaning, stop. Else print the result of calling '1'. Then, unless
+# we are in -n mode, invoke the result of calling '1' in the shell
+$(if $(filter clean,$(MAKECMDGOALS)),,$(eval S2S += $$(info $$(call $$1,$$2,$$3)))\
+$(if $(filter n,$(MAKEFLAGS)),,$(eval S2S += $$(shell $$(call $$1,$$2,$$3)))))
+# Rules
.SECONDEXPANSION:
-apc-d apc: $$($$@SRC) | $(hGEN)
- $(strip $(LDCMD))
+$(DRIVERS:%=%-d) $(DRIVERS): $$($$@SRC) | $(hGEN)
+ $(strip $(call LDCMD,$^,$@))
%-d.o: CFLAGS+= -Og -ggdb
%.o %-d.o: %.c $$(call cGENDEP,$$(dir $$@)%.c)
- $(strip $(CCMD))
-
-%.tab.h %.tab.c: %.y
- $(strip $(YCMD))
- mv $(notdir $(<:%.y=%.tab.[ch])) $(dir $@)
+ $(call CCMD,$<,$@)
-%.c: %.rl
- $(strip $(RLCMD))
+%.tab.h: %.tab.c ;
+%.tab.c: %.y $$(call S2S,YCMD,%.y,$$@) ;
+%.c: %.rl $$(call S2S,RLCMD,%.rl,$$@) ;
-clean: $(wildcard $(cGEN) $(hGEN) $(apcSRC) $(apc-dSRC))
- rm $^
+clean: $(wildcard $(cGEN) $(hGEN) $(foreach drv,$(DRIVERS),$($(drv)SRC) $($(drv)-dSRC)))
+ $(if $^,rm $^)