Вы можете создать свою собственную команду make для компиляции и использовать ее в своих рецептах.Например, вы можете автоматически записывать стандартные и выходные данные об ошибках вашего компилятора в текстовые файлы и grep
все они для предупреждений в конце вашей сборки.Что-то вроде (с GNU make):
# $(1): source file
# $(2): object file
# $(3): log file
MYCXX = $(CXX) $(CXXFLAGS) -c $(1) -o $(2) 2>&1 | tee $(3)
CXXFLAGS += -Wall -pedantic
%.o: %.cpp
$(call MYCXX,$<,$@,$*.log)
OBJS := $(patsubst %.cpp,%.o,$(wildcard *.cpp))
LOGS := $(patsubst %.o,%.log,$(OBJS))
top: $(OBJS)
$(CXX) $(LDFLAGS) $^ -o $@
@printf '\nSummary of warnings\n###################\n\n'
@for l in $(LOGS); do grep -i -10 warning $$l || true; done
Демонстрация:
$ make
g++ -Wall -pedantic -c b.cpp -o b.o 2>&1 | tee b.log
g++ -Wall -pedantic -c c.cpp -o c.o 2>&1 | tee c.log
g++ -Wall -pedantic -c a.cpp -o a.o 2>&1 | tee a.log
g++ -Wall -pedantic -c d.cpp -o d.o 2>&1 | tee d.log
c.cpp: In function ‘int c()’:
c.cpp:1:18: warning: unused variable ‘vc’ [-Wunused-variable]
int c(void) {int vc; return 0;}
^~
a.cpp: In function ‘int a()’:
a.cpp:1:18: warning: unused variable ‘va’ [-Wunused-variable]
int a(void) {int va; return 0;}
^~
b.cpp: In function ‘int b()’:
b.cpp:1:18: warning: unused variable ‘vb’ [-Wunused-variable]
int b(void) {int vb; return 0;}
^~
g++ b.o c.o a.o d.o -o top
Summary of warnings
###################
b.cpp: In function ‘int b()’:
b.cpp:1:18: warning: unused variable ‘vb’ [-Wunused-variable]
int b(void) {int vb; return 0;}
^~
c.cpp: In function ‘int c()’:
c.cpp:1:18: warning: unused variable ‘vc’ [-Wunused-variable]
int c(void) {int vc; return 0;}
^~
a.cpp: In function ‘int a()’:
a.cpp:1:18: warning: unused variable ‘va’ [-Wunused-variable]
int a(void) {int va; return 0;}
^~
Существует даже немного более элегантный способ, заключающийся в (пере) определении стандартного make CXX
переменной и назначьте ему небольшой сценарий оболочки, который выполняет эту работу, так что вы можете использовать CXX
, как обычно:
CXX = function mycxx { g++ $$* 2>&1 | tee $(patsubst %.o,%.log,$@) ;}; mycxx
%.o: %.cpp
$(CXX) $(CXXFLAGS) -c $< -o $@
При расширении make для цели foo.o
рецепт становится:
function mycxx { g++ $* 2>&1 | tee foo.log ;}; mycxx -Wall -pedantic foo.cpp -o foo.o
И оболочка, наконец, выполнится:
g++ -Wall -pedantic foo.cpp -o foo.o 2>&1 | tee foo.log
Окончательный результат должен быть таким же, как и в первом решении, но со значительно меньшими изменениями вашего Makefile.Помимо (пере) определения CXX
, оно должно остаться прежним (при условии, что вы использовали CXX
в своих рецептах, что рекомендуется).