X-Git-Url: https://www.kengrimes.com/gitweb/?p=henge%2Fapc.git;a=blobdiff_plain;f=Makefile;h=7b8934d73c96e2b4ba774231e97417d50c2cb92d;hp=d6689db596e7ce86fca439a10236e170afdaef10;hb=HEAD;hpb=dbce0e5aa9599ae86cb1aa33966b09b9dad3e810 diff --git a/Makefile b/Makefile index d6689db..7b8934d 100644 --- a/Makefile +++ b/Makefile @@ -5,37 +5,86 @@ ################################################################################ # This makefile builds APC, the Asset Package Compiler for Henge, on the system. ################################################################################ -# Directories (can override with env vars) -APC_ROOTDIR := $(if $(APC_ROOTDIR),$(APC_ROOTDIR),.) -APC_SRCDIR := $(if $(APC_SRCDIR),$(APC_SRCDIR),$(APC_ROOTDIR)/src) +# Driver sources +DRIVERS ?= apc testapc testston -# APC is built from Ragel, Bison and C source code only. -ySRC := $(shell find $(APC_SRCDIR) -type f -name '*.y') -rlSRC := $(shell find $(APC_SRCDIR) -type f -name '*.rl') -cSRC := $(shell find $(APC_SRCDIR) -type f -name '*.c') +# Debug Level +DEBUG ?= 1 + +# Yacc +YACC := bison +YFLAGS ?= -v -d -Wall +YCMD = $(strip $(YACC) $(YFLAGS) $(if $2,$(dir $2))$1) +YCMD += $(if $2,&& mv $(notdir $(1:%.y=%.tab.[ch])) $(dir $2)) + +# Ragel +RLC ?= ragel +RLFLAGS ?= -C +RLCMD = $(strip $(RLC) $(RLFLAGS) $(if $2,-o $2 $(dir $2))$1) + +# C +CC ?= gcc +CFLAGS ?= -Wall +CCMD = $(strip $(CC) $(CFLAGS) $(CPPFLAGS) -c $(if $2,-o $2) $1) + +# Linker +LD ?= ld +LDFLAGS ?= +LDLIBS ?= -lunistring +apcLIBS ?= +apc-dLIBS ?= +LDCMD = $(strip $(CC) $(LDFLAGS) $(if $2,-o $2) $1) $(LDLIBS) $($1LIBS) -# Specify the linker (allow env var override) -LD := $(if $(LD),$(LD),ld) +# APC is built from Ragel, Bison and C source code only. +ySRC := $(shell find src/ -type f -name '*.y') +rlSRC := $(shell find src/ -type f -name '*.rl') +cSRC := $(shell find src/ -type f -name '*.c') # Generated files from Yacc/Bison and Ragel hGEN := $(ySRC:%.y=%.tab.h) -cGEN := $(strip $(ySRC:%.y=%.tab.c) $(rlSRC:%.rl=%.c)) +cGEN := $(strip $(ySRC:%.y=%.tab.c) $(rlSRC:%.rl=%.fsm.c)) +.SECONDARY: $(cGEN) -# Rules -apc-d: $(patsubst %.c,%-d.o,$(cSRC) $(cGEN)) | $(hGEN) - $(LD) $(LDFLAGS) $(LDLIBS) $< +# Determine binary/ir targets (object files and driver binaries) +cTRG := $(patsubst %.c,%.o,$(cSRC) $(cGEN)) +ldSRC := $(filter-out $(DRIVERS:%=\%/%.o),$(cTRG)) +cTRG += $(cTRG:%.o=%-d.o) +ldTRG := $(DRIVERS:%=%-d) $(DRIVERS) +ldDEP = $(filter %/$1.o,$(cTRG)) +ldDEP += $(if $(filter testston%,$1),,$(if $(filter %-d,$1),$(ldSRC:%.o=%-d.o),$(ldSRC))) -apc: $(patsubst %.c,%.o,$(cSRC) $(cGEN)) | $(hGEN) - $(LD) $(LDFLAGS) $(LDLIBS) $< +# Determine if '1' is newer than '2' +TSTAMP = $(if $(wildcard $1),$(shell stat -c %Y $1),) +NEWER = $(eval 4 := $(call TSTAMP,$(dir $2)$1)) +NEWER += $(eval 5 := $(call TSTAMP,$2)) +NEWER += $(if $(filter $5,$(firstword $(sort $4 $5))),$1,$2) -%.tab.h %.tab.c: %.y - bison -d $(YFLAGS) $< +ifeq (,$(filter clean,$(MAKECMDGOALS))) +# Deps should be generated for each source file, when not cleaning +cGENDEP = $(if $(wildcard $1),$(subst $(dir $1),,$(filter-out $1 \ %:,$(shell $(CC) -MM -MG $1))),\ +$(info [<$1>: no deps - file not found])) +# Filter only missing deps +cMISDEP = $(strip $(foreach dep,$(call cGENDEP,$1),$(if $(wildcard src/$(dep)),,$(dep)))) +# S2S will print the command necessary to create a file when called +S2S = $(if $(filter $2,$(call NEWER,$2,$3)),$(eval 4 := t),$(eval 4 :=)) +S2S += $(if $4,$(info $(call $1,$2,$3))) +ifeq (,$(filter n,$(MAKEFLAGS))) +# Unless we're in -n mode, S2S should also invoke the command on the shell +S2S += $(if $4,$(shell $(call $1,$2,$3))) +endif +endif -cFLAGS = $(strip $(CFLAGS) $(CPPFLAGS) $1 $2 $3 $4 $5 $6 $7 $8 $9) -GENDEP = $(filter-out \ %:,$(shell $(CC) -MM -MG $1)) -.SECONDEXPANSION: -%.o: %.c $$(call GENDEP,$$(dir $$@)%.c) - $(CC) $(call cFLAGS,-c,-o $@) $< +# Clean targets +cleanCMD = $(if $(wildcard $1),rm $(wildcard $(sort $1))) -%-d.o: $$(call GENDEP,$$(dir $$@)%.c) $$(info [DEPS|$$@] $$(call GENDEP,$$(dir $$@)%.c)) - $(CC) $(call cFLAGS,-c,-Og,-ggdb,-o $@) $< +# Rules +.SECONDEXPANSION: +$(ldTRG): $$(call ldDEP,$$@) | $(hGEN) ; $(call LDCMD,$^,$@) +%-d.o: CFLAGS+= -Og -ggdb -DDEBUG=$(DEBUG) +%.o %-d.o: %.c $$(call cGENDEP,$$(dir $$@)%.c); $(call CCMD,$<,$@) +%.o %-d.o: %.c; $(error Missing dependencies for $<: $(call cMISDEP, $<)) +%.tab.h: %.tab.c ; +%.tab.c: %.y $$(call S2S,YCMD,%.y,$$@) ; +%.fsm.c: %.rl $$(call S2S,RLCMD,%.rl,$$@) ; +clean: ; $(call cleanCMD,$(cGEN) $(hGEN) $(cTRG)) +distclean: clean ; $(call cleanCMD,$(ldTRG))