mirror of
				https://github.com/eledio-devices/thirdparty-littlefs.git
				synced 2025-10-31 16:14:16 +01:00 
			
		
		
		
	GCC is a bit frustrating here, it really wants to generate every file in a single command, which _is_ more efficient if our build system could leverage this. But -fcallgraph-info is a rather novel flag, so we can't really rely on it for generally compiling and testing littlefs. The multi-file output gets in the way when we want an explicitly separate rule for callgraph-info generation. We can't generate the callgraph-info without generating the objects files. This becomes a surprsing issue when parallel building (make -j) is used! Suddenly we might end up with both the .o and .ci rules writing to .o files, which creates a really difficult to track down issue of corrupted .o files. The temporary solution is to use an order-only prerequisite. This still ends up building the .o files twice, but it's an acceptable tradeoff for not requiring the -fcallgraph-info for all builds.
		
			
				
	
	
		
			173 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Makefile
		
	
	
	
	
	
			
		
		
	
	
			173 lines
		
	
	
		
			3.9 KiB
		
	
	
	
		
			Makefile
		
	
	
	
	
	
| ifdef BUILDDIR
 | |
| # make sure BUILDDIR ends with a slash
 | |
| override BUILDDIR := $(BUILDDIR)/
 | |
| # bit of a hack, but we want to make sure BUILDDIR directory structure
 | |
| # is correct before any commands
 | |
| $(if $(findstring n,$(MAKEFLAGS)),, $(shell mkdir -p \
 | |
| 	$(BUILDDIR) \
 | |
| 	$(BUILDDIR)bd \
 | |
| 	$(BUILDDIR)tests))
 | |
| endif
 | |
| 
 | |
| # overridable target/src/tools/flags/etc
 | |
| ifneq ($(wildcard test.c main.c),)
 | |
| TARGET ?= $(BUILDDIR)lfs
 | |
| else
 | |
| TARGET ?= $(BUILDDIR)lfs.a
 | |
| endif
 | |
| 
 | |
| 
 | |
| CC      ?= gcc
 | |
| AR      ?= ar
 | |
| SIZE    ?= size
 | |
| CTAGS   ?= ctags
 | |
| NM      ?= nm
 | |
| OBJDUMP ?= objdump
 | |
| LCOV    ?= lcov
 | |
| 
 | |
| SRC ?= $(wildcard *.c)
 | |
| OBJ := $(SRC:%.c=$(BUILDDIR)%.o)
 | |
| DEP := $(SRC:%.c=$(BUILDDIR)%.d)
 | |
| ASM := $(SRC:%.c=$(BUILDDIR)%.s)
 | |
| CGI := $(SRC:%.c=$(BUILDDIR)%.ci)
 | |
| 
 | |
| ifdef DEBUG
 | |
| override CFLAGS += -O0
 | |
| else
 | |
| override CFLAGS += -Os
 | |
| endif
 | |
| ifdef TRACE
 | |
| override CFLAGS += -DLFS_YES_TRACE
 | |
| endif
 | |
| override CFLAGS += -g3
 | |
| override CFLAGS += -I.
 | |
| override CFLAGS += -std=c99 -Wall -pedantic
 | |
| override CFLAGS += -Wextra -Wshadow -Wjump-misses-init -Wundef
 | |
| 
 | |
| ifdef VERBOSE
 | |
| override TESTFLAGS     += -v
 | |
| override CALLSFLAGS    += -v
 | |
| override CODEFLAGS     += -v
 | |
| override DATAFLAGS     += -v
 | |
| override STACKFLAGS    += -v
 | |
| override STRUCTSFLAGS  += -v
 | |
| override COVERAGEFLAGS += -v
 | |
| endif
 | |
| ifdef EXEC
 | |
| override TESTFLAGS += --exec="$(EXEC)"
 | |
| endif
 | |
| ifdef COVERAGE
 | |
| override TESTFLAGS += --coverage
 | |
| endif
 | |
| ifdef BUILDDIR
 | |
| override TESTFLAGS     += --build-dir="$(BUILDDIR:/=)"
 | |
| override CALLSFLAGS    += --build-dir="$(BUILDDIR:/=)"
 | |
| override CODEFLAGS     += --build-dir="$(BUILDDIR:/=)"
 | |
| override DATAFLAGS     += --build-dir="$(BUILDDIR:/=)"
 | |
| override STACKFLAGS    += --build-dir="$(BUILDDIR:/=)"
 | |
| override STRUCTSFLAGS  += --build-dir="$(BUILDDIR:/=)"
 | |
| override COVERAGEFLAGS += --build-dir="$(BUILDDIR:/=)"
 | |
| endif
 | |
| ifneq ($(NM),nm)
 | |
| override CODEFLAGS += --nm-tool="$(NM)"
 | |
| override DATAFLAGS += --nm-tool="$(NM)"
 | |
| endif
 | |
| ifneq ($(OBJDUMP),objdump)
 | |
| override STRUCTSFLAGS += --objdump-tool="$(OBJDUMP)"
 | |
| endif
 | |
| 
 | |
| 
 | |
| # commands
 | |
| .PHONY: all build
 | |
| all build: $(TARGET)
 | |
| 
 | |
| .PHONY: asm
 | |
| asm: $(ASM)
 | |
| 
 | |
| .PHONY: size
 | |
| size: $(OBJ)
 | |
| 	$(SIZE) -t $^
 | |
| 
 | |
| .PHONY: tags
 | |
| tags:
 | |
| 	$(CTAGS) --totals --c-types=+p $(shell find -H -name '*.h') $(SRC)
 | |
| 
 | |
| .PHONY: calls
 | |
| calls: $(CGI)
 | |
| 	./scripts/calls.py $^ $(CALLSFLAGS)
 | |
| 
 | |
| .PHONY: test
 | |
| test:
 | |
| 	./scripts/test.py $(TESTFLAGS)
 | |
| .SECONDEXPANSION:
 | |
| test%: tests/test$$(firstword $$(subst \#, ,%)).toml
 | |
| 	./scripts/test.py $@ $(TESTFLAGS)
 | |
| 
 | |
| .PHONY: code
 | |
| code: $(OBJ)
 | |
| 	./scripts/code.py $^ -S $(CODEFLAGS)
 | |
| 
 | |
| .PHONY: data
 | |
| data: $(OBJ)
 | |
| 	./scripts/data.py $^ -S $(DATAFLAGS)
 | |
| 
 | |
| .PHONY: stack
 | |
| stack: $(CGI)
 | |
| 	./scripts/stack.py $^ -S $(STACKFLAGS)
 | |
| 
 | |
| .PHONY: structs
 | |
| structs: $(OBJ)
 | |
| 	./scripts/structs.py $^ -S $(STRUCTSFLAGS)
 | |
| 
 | |
| .PHONY: coverage
 | |
| coverage:
 | |
| 	./scripts/coverage.py $(BUILDDIR)tests/*.toml.info -s $(COVERAGEFLAGS)
 | |
| 
 | |
| .PHONY: summary
 | |
| summary: $(BUILDDIR)lfs.csv
 | |
| 	./scripts/summary.py -Y $^ $(SUMMARYFLAGS)
 | |
| 
 | |
| 
 | |
| # rules
 | |
| -include $(DEP)
 | |
| .SUFFIXES:
 | |
| 
 | |
| $(BUILDDIR)lfs: $(OBJ)
 | |
| 	$(CC) $(CFLAGS) $^ $(LFLAGS) -o $@
 | |
| 
 | |
| $(BUILDDIR)lfs.a: $(OBJ)
 | |
| 	$(AR) rcs $@ $^
 | |
| 
 | |
| $(BUILDDIR)lfs.csv: $(OBJ) $(CGI)
 | |
| 	./scripts/code.py $(OBJ) -q $(CODEFLAGS) -o $@
 | |
| 	./scripts/data.py $(OBJ) -q -m $@ $(DATAFLAGS) -o $@
 | |
| 	./scripts/stack.py $(CGI) -q -m $@ $(STACKFLAGS) -o $@
 | |
| 	./scripts/structs.py $(OBJ) -q -m $@ $(STRUCTSFLAGS) -o $@
 | |
| 	$(if $(COVERAGE),\
 | |
| 		./scripts/coverage.py $(BUILDDIR)tests/*.toml.info \
 | |
| 			-q -m $@ $(COVERAGEFLAGS) -o $@)
 | |
| 
 | |
| $(BUILDDIR)%.o: %.c
 | |
| 	$(CC) -c -MMD $(CFLAGS) $< -o $@
 | |
| 
 | |
| $(BUILDDIR)%.s: %.c
 | |
| 	$(CC) -S $(CFLAGS) $< -o $@
 | |
| 
 | |
| # gcc depends on the output file for intermediate file names, so
 | |
| # we can't omit to .o output. We also need to serialize with the
 | |
| # normal .o rule because otherwise we can end up with multiprocess
 | |
| # problems with two instances of gcc modifying the same .o
 | |
| $(BUILDDIR)%.ci: %.c | $(BUILDDIR)%.o
 | |
| 	$(CC) -c -MMD -fcallgraph-info=su $(CFLAGS) $< -o $|
 | |
| 
 | |
| # clean everything
 | |
| .PHONY: clean
 | |
| clean:
 | |
| 	rm -f $(TARGET)
 | |
| 	rm -f $(TARGET).*.csv
 | |
| 	rm -f $(OBJ)
 | |
| 	rm -f $(CGI)
 | |
| 	rm -f $(DEP)
 | |
| 	rm -f $(ASM)
 | |
| 	rm -f $(BUILDDIR)tests/*.toml.*
 |