我有一个脚本生成一个version.h
文件,其中包含有关特定构建的各种详细信息
该脚本运行各种命令,例如:
readonly VERSION=$(git describe --always --dirty --long --tags)
readonly NUM_COMMITS=$(git rev-list HEAD | wc -l | bc)
readonly BRANCH=$(git rev-parse --abbrev-ref HEAD)
readonly AHEAD_BY=$(git log --oneline origin/${BRANCH}..${BRANCH} | wc -l | bc)
readonly NUM_UNTRACKED=$(git ls-files --exclude-standard --others --full-name -- . | wc -l | bc)
readonly HOSTNAME=$(hostname)
然后写入一个临时文件,检查预先存在的生成文件,如果它们不同,则用新文件覆盖旧version.h
。
#pragma once
/*
* Version information
*
* - This is a generated file - do not edit
*/
namespace foo { namespace app {
static const char VERSION[] = "26b75cd";
static const char NUM_COMMITS[] = "224";
static const char BRANCH[] = "master";
static const char AHEAD_BY[] = "0";
static const char NUM_UNTRACKED[] = "0";
static const char USER[] = "steve";
static const char HOSTNAME[] = "steve-linux";
static const char BUILD_VARIANT[] = "debug";
static const char BUILD_DATE[] = __DATE__ " " __TIME__;
}}
当文件更新时,它也会打印到stdout
version updated: 26b75cd
版本脚本应该是第一个运行的东西,应该在每次调用makefile时运行。
目前,我通过在makefile
中使用简单扩展变量来实现这一目标new_ver := $(shell ./app/gen_version.sh $(BUILD))
它有效,但脚本的任何输出都在new_var
中捕获,我无法显示该输出。
这是一个可接受的权衡(不显示新版本),但在理想的世界中,我想显示脚本输出。
我不确定添加到默认.PHONY
目标的all
目标是否有效,因为每个应用都有自己的目标,因此您可以构建它。
define do-make-bin
binaries += $(addprefix $(BIN_DIR),$1)
bin_sources += $2
# a target so users can call 'make <bin-name>' and build only the bin and its dependencies
$1: $(addprefix $(BIN_DIR),$1)
# the target to build the actual binary
$(addprefix $(BIN_DIR),$1): $(call src_to_obj,$2) $(addsuffix .so,$(addprefix $(LIB_DIR)lib,$3)) $(addsuffix .a,$(addprefix $(LIB_DIR)lib,$4))
@$(CXX) $(call src_to_obj,$2) -L/usr/lib -L$(SDK_LIB_DIR) -L$(LIB_DIR) $(TARGET_LINK_FLAGS) -Wl$(,)-Bstatic $(addprefix -l,$4) $(addprefix -l,$6) -Wl$(,)-Bdynamic -Wl$(,)-rpath$(,)$(RPATH) $(linkpath) -L$(LIB_DIR) $(sdk_libs) $(tcmalloc_libs) -lpthread -lrt $(addprefix -l,$3) $(addprefix -l,$5) -rdynamic -o $$@
endef
调用make foo
仍应检查version.h
是否需要更新
答案 0 :(得分:1)
当目标没有先决条件且存在时,文件没有任何反应。您需要的是在创建临时文件时创建条件先决条件。您可以使用gnu / make的wildcard
功能。
如果调用临时文件version.new.h
并假设脚本检查version.h
和version.new.h
之间的差异。如果它们相同则删除它。然后你可以使用类似的东西:
version.h : $(wildcard version.new.h)
mv -f version.new.h version.h
为了验证是否首先创建了version.h,您应该分割作业。
然后你确保第一份工作先运行。 Makefile
代码的示例是:
all: version_h other_staff
.PHONY: version_h
version_h:
./app/gen_version.sh $(BUILD)
$(MAKE) version.h
version.h : $(wildcard version.new.h)
mv -f version.new.h version.h
other_staff: $(EXE) $(LIBS)
答案 1 :(得分:1)
确保它在每个目标之前运行的唯一方法是,如果将自己局限于普通的makefile方法,就是将内容放入配方中,然后使构建该内容的目标成为每个目标的先决条件< / em>在你的makefile中。虽然这个可以以这样的方式完成,它不会导致每次都重建,但这很糟糕。另外,它不会让你把输出放到像new_ver
这样的变量中(你说你想要显示它,但你也没有说明你是否也需要它)。
但是,如果唯一的问题是您没有显示new_ver
的值,那么 就会轻易解决:
new_ver := $(shell ./app/gen_version.sh $(BUILD))
$(info $(new_ver))