给基于cmake的工程添加uninstall功能

  我们知道,cmake提供了install指令可以让我们在项目构建完成后,通过make install命令,或者通过cmake --install . --prefix=<安装目录> 命令的方式来将CMakeLists.txt文件中通过install指令配置的文件安装到目标目录中。

  但是,cmake却没有提供uninstall指令来移除这些安装进去的文件的功能,这给工程的卸载添加了一些麻烦。为了实现卸载功能,我们只能自己来编写CMake脚本来实现。

  为了实现卸载功能,我们首先需要知道cmake在部署的时候在目标目录安装进去了哪些文件。幸好,cmake本身已经为我们准备了一个文件叫做install_manifest.txt,它包含了所有通过install指令安装的文件列表,每个文件占用一行。只要我们通过make install安装过,我们就可以在cmake的当前工作目录下面找到这个文件。

  因此,我们就可以利用这个install_manifest.txt文件来执行删除文件的操作,从而来达成卸载安装文件的目的。当然,我们可以用一个shell脚本只要少数几行代码来实现,但是,这里还是基于cmake,让用户能够已约定俗成的习惯方式通过make uninstall的方式来进行卸载操作。

  首先创建一个Uninstall.cmake.in 的文件,该文件和CMakeLists.txt放在一个目录下面,如下:


# CMAKE_BINARY_DIR变量指向cmake build tree的顶级目录。
if(NOT EXISTS "@CMAKE_BINARY_DIR@/install_manifest.txt")
  message(FATAL_ERROR "Cannot find install manifest: @CMAKE_BINARY_DIR@/install_manifest.txt")
endif()

# 从install_manifest.txt中读取文本内容,并根据\n拆分成一个文件列表
file(READ "@CMAKE_BINARY_DIR@/install_manifest.txt" files)
string(REGEX REPLACE "\n" ";" files "${files}")

# 循环删除列表中的文件
foreach(file ${files})
  message(STATUS "Uninstalling $ENV{DESTDIR}${file}")
  
  if(IS_SYMLINK "$ENV{DESTDIR}${file}" OR EXISTS "$ENV{DESTDIR}${file}")
  
    # 调用cmake -E remove <文件名>来删除文件
    # 其中$ENV{DESTDIR}当make DESTDIR=<安装目录> install
    # 在命令行指定安装目录的将被设置,但是install_manifest.txt中
    # 的文件路径不包括DESTDIR指定的部分
    exec_program(
      "@CMAKE_COMMAND@" ARGS "-E remove \"$ENV{DESTDIR}${file}\""
      OUTPUT_VARIABLE rm_out
      RETURN_VALUE rm_retval
      )
      
    if(NOT "${rm_retval}" STREQUAL 0)
      message(FATAL_ERROR "Problem when removing $ENV{DESTDIR}${file}")
    endif()
    
  else(IS_SYMLINK "$ENV{DESTDIR}${file}" OR EXISTS "$ENV{DESTDIR}${file}")
    message(STATUS "File $ENV{DESTDIR}${file} does not exist.")
  endif()
endforeach()

  然后在CMakeLists.txt文件中添加下面代码,如下:


# 创建卸载target
if(NOT TARGET uninstall)

  # 利用前面的Uninstall.cmake.in文件自动生成Uninstall.cmake
  configure_file(
    "${CMAKE_CURRENT_SOURCE_DIR}/Uninstall.cmake.in"
    "${CMAKE_CURRENT_BINARY_DIR}/Uninstall.cmake"
    IMMEDIATE @ONLY)
  
  # 通过add_custom_target创建一个自定义构建目标的命令
  # 自定义目标可以在构建时被执行,但不会产生实际的构建输出。
  add_custom_target(uninstall
    COMMAND ${CMAKE_COMMAND} -P ${CMAKE_CURRENT_BINARY_DIR}/Uninstall.cmake)

endif()
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

码农心语

您的鼓励是我写作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值