makefile-cmake 打印 git commit 等编译信息

1. makefile-cmake 打印 git commit 等编译信息

本方法记录如何在程序中打印版本信息,比如 git commit id, git branch, git tag, 编译时间,编译用户等

1.1. 前置条件及目的

目的:在编译时,记录 git commit 信息,打印到日志中,方便调试定位问题

举例:

目录结构如下

├── build.sh
├── CMakeLists.txt
├── config.h.in
├── makefile
└── src
    ├── CMakeLists.txt
    └── main.c

先定义 config.h.in 文件,里面定义 SDK_PRJ_USER,SDK_PRJ_TIME,SDK_PRJ_COMMIT 等变量。

#ifndef _SDK_CONFIG_H_
#define _SDK_CONFIG_H_

#define SDK_BUILD_USER "@SDK_BUILD_USER@"

#define SDK_BUILD_TIME "@SDK_BUILD_TIME@"

#define SDK_BUILD_COMMIT "@SDK_BUILD_COMMIT@"

#endif /* _SDK_CONFIG_H_ */

在 main.c 里包含 config.h 文件,然后打印这些变量。

#include <stdio.h>
#include "config.h"

int main(int argc, char *argv[]) {
    printf("time %s, user %s, commit %s\n", SDK_BUILD_TIME, SDK_BUILD_USER, SDK_BUILD_COMMIT);
    return 0;
}

1.2. shell 通用方案

  • 在真正的编译之前,通过 shell 来修改 config.h.in 文件,然后编译。
  • 当然也可以在 shell 中调用编译命令
  • 依赖 sed git 等命令,window 环境需要安装 sed 命令,git自带 sed 命令,将 C:\Program Files\Git\usr\bin 加入环境变量即可

build.sh 文件内容如下:

# project info
SDK_PRJ_USER=`git config user.email`
SDK_PRJ_TIME=`date +%Y%m%d_%H%M%S`
SDK_PRJ_COMMIT=`git rev-parse --short HEAD`

cp config.h.in config.h
sed -i 's/#define SDK_BUILD_TIME.*/#define SDK_BUILD_TIME "'${SDK_PRJ_TIME}'"/g' config.h
sed -i 's/#define SDK_BUILD_USER.*/#define SDK_BUILD_USER "'${SDK_PRJ_USER}'"/g' config.h
sed -i 's/#define SDK_BUILD_COMMIT.*/#define SDK_BUILD_COMMIT "'${SDK_PRJ_COMMIT}'"/g' config.h

gcc src/main.c -I./

1.3. Makefile 方案

  • 类似于 shell 方案,依赖 sed git 等命令

makefile 文件内容如下:

SDK_PRJ_USER=$(shell git config user.email)
SDK_PRJ_TIME=$(shell date +%Y%m%d_%H%M%S)
SDK_PRJ_COMMIT=$(shell git rev-parse --short HEAD)
	
all: config demo

config:
	cp config.h.in config.h
	sed -i 's/#define SDK_BUILD_TIME.*/#define SDK_BUILD_TIME "'$(SDK_PRJ_TIME)'"/g' config.h
	sed -i 's/#define SDK_BUILD_USER.*/#define SDK_BUILD_USER "'$(SDK_PRJ_USER)'"/g' config.h
	sed -i 's/#define SDK_BUILD_COMMIT.*/#define SDK_BUILD_COMMIT "'$(SDK_PRJ_COMMIT)'"/g' config.h

demo: $(OBJ1)
	gcc src/main.c -I./

clean:
	rm -f a.out config.h

1.4. CMakeLists.txt 方案

  • 定义了 git 获取结果的宏,然后调用宏
  • 最顶层 CMakeLists.txt 文件
cmake_minimum_required(VERSION 3.5)

PROJECT (DEMO)
SET(EXECUTABLE_OUTPUT_PATH ${PROJECT_BINARY_DIR}/bin)

macro(git_cmd_result _git_result)               # 宏的开始
    find_package(Git QUIET)                     # 查找Git,QUIET静默方式不报错
    if(GIT_FOUND)
      execute_process(                          # 执行一个子进程
          COMMAND ${GIT_EXECUTABLE} ${git_cmd}  # 命令,需要预定义 git_cmd 变量,制定 git 命令
          OUTPUT_VARIABLE ${_git_result}        # 输出字符串存入变量
          OUTPUT_STRIP_TRAILING_WHITESPACE      # 删除字符串尾的换行符
          ERROR_QUIET                           # 对执行错误静默
          WORKING_DIRECTORY                     # 执行路径
          ${CMAKE_CURRENT_SOURCE_DIR}
        )
    endif()
endmacro()                      # 宏的结束

set(git_cmd rev-parse --short HEAD)  # 查找 commit id
git_cmd_result(SDK_BUILD_COMMIT)

set(git_cmd config user.email)       # 查找 git user email
git_cmd_result(SDK_BUILD_USER)

set(SDK_BUILD_TIME "")
string(TIMESTAMP SDK_BUILD_TIME %Y%m%d_%H%M%S)

# 配置文件
configure_file(${PROJECT_SOURCE_DIR}/config.h.in ${PROJECT_BINARY_DIR}/include/config.h)

ADD_SUBDIRECTORY(src)
  • src/CMakeLists.txt 文件
MESSAGE(STATUS "This is BINARY dir " ${PROJECT_BINARY_DIR})

INCLUDE_DIRECTORIES(${PROJECT_BINARY_DIR}/include)
ADD_EXECUTABLE(demo main.c)
  • 5
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值