如何使用CMake execute_process()正确调用MSYS2 Bash命令?

时间:2016-02-27 08:09:11

标签: bash shell cmake mingw-w64 msys2

问题描述

我在设置执行MSYS2 bash命令的CMake external_process()命令时遇到问题。当我在MSYS2 shell中运行命令$ bash -v ./bootstrap.sh时,该命令正常工作。但是如果我使用$ cmake -P Run_bash_command.cmake在MSYS2 shell中运行CMake脚本,那么命令会在整个过程中出错。我在CMake Documentation中找到的一条重要信息让我觉得我没有正确地调用bash或者错过了一个环境变量:

  

CMake直接使用操作系统API执行子进程。所有参数都将VERBATIM传递给子进程。没有使用中间shell,因此shell运算符如>被视为正常的论点。

如果可能的话,我希望能够使用CMake执行此命令,因为此问题是更大的CMake superbuild项目的一部分。如果有另一种解决问题的方法,只要我可以将其包含在superbuild项目的自动化中,我就可以接受建议。任何帮助将不胜感激。

Run_bash_command.cmake内容:

SET( ENV{MSYSTEM} MINGW64 )
SET( DIR_CONTAINING_BOOTSTRAP_SH C:/bash_test )
SET( BASH_COMMAND_TO_RUN bash -v ./bootstrap.sh )

EXECUTE_PROCESS( COMMAND ${BASH_COMMAND_TO_RUN}
          WORKING_DIRECTORY ${DIR_CONTAINING_BOOTSTRAP_SH} RESULT_VARIABLE command_result )

IF( NOT "${command_result}" STREQUAL "0" )
    MESSAGE( FATAL_ERROR "Error: command_result='${command_result}'" )
ENDIF()

环境设置

  • 我按照setup MSYS2 64bit的指示,使用命令pacman -S base-devel git mingw-w64-x86_64-cmake mingw-w64-x86_64-toolchain
  • 添加了mingw-w64工具链以及cmake
  • 要运行命令,我使用随MSYS2一起安装的 MinGW-w64 Win64 Shell
  • 文件 bootstrap.sh 来自libusb github repository,文件夹 c:/ bash_test 包含主分支的克隆

输出

$ bash -v ./bootstrap.sh输出:

$ bash -v ./bootstrap.sh
#!/bin/sh

if ! test -d m4 ; then
    mkdir m4
fi
autoreconf -ivf || exit 1
autoreconf: Entering directory `.'
autoreconf: configure.ac: not using Gettext
autoreconf: running: aclocal --force -I m4
autoreconf: configure.ac: tracing
autoreconf: running: libtoolize --copy --force
libtoolize: putting auxiliary files in '.'.
libtoolize: copying file './ltmain.sh''
...<clipped output due to length>...
configure.ac:29: installing './install-sh'
configure.ac:29: installing './missing'
examples/Makefile.am: installing './depcomp'
autoreconf: Leaving directory `.'

$ cmake -P Run_bash_command.cmake输出:

$ cmake -P Run_bash_command.cmake
#!/bin/sh

if ! test -d m4 ; then
    mkdir m4
fi
autoreconf -ivf || exit 1
autoreconf: Entering directory `.'
autoreconf: configure.ac: not using Gettext
autoreconf: running: aclocal --force -I m4
aclocal-1.15: error: aclocal: file '/msys64/usr/share/aclocal/xsize.m4' does not exist
autoreconf: aclocal failed with exit status: 1
CMake Error at Run_bash_command.cmake:10 (MESSAGE):
  Error: command_result='1'

我尝试过的事情:

  • 替换bash -l -c但这会导致shell默认为主目录,然后无法找到文件bootstrap.sh
  • 通过检查我的环境路径变量
  • 验证了正确的bash版本
  • 已验证的MSYS2及其软件包是最新的
  • 使用sh代替bash
  • 直接致电autoreconf -ivf但发生同样的问题
  • 使用Unix样式路径而不是Windows样式

1 个答案:

答案 0 :(得分:1)

我能够使用下面的代码解决问题。

Run_bash_command.cmake内容:

SET( DIR_CONTAINING_BOOTSTRAP_SH /C/bash_test )
SET( BASH_COMMAND_TO_RUN bash -l -c "cd ${DIR_CONTAINING_BOOTSTRAP_SH} && sh ./bootstrap.sh" )

EXECUTE_PROCESS( COMMAND ${BASH_COMMAND_TO_RUN}
                WORKING_DIRECTORY ${DIR_CONTAINING_BOOTSTRAP_SH} RESULT_VARIABLE command_result )

IF( NOT "${command_result}" STREQUAL "0" )
    MESSAGE( FATAL_ERROR "Error: command_result='${command_result}'" )
ENDIF()

重要说明:

  • 使用bash -l会导致shell默认为主目录,为了解决这个问题,我添加了一个更改目录cd <path>命令,让我们回到我们想要发出bash命令的目录。
  • 使用-c会导致bash从字符串中读取命令。由于我们要发出两个命令,一个用于更改目录,另一个用于运行shell脚本,我们需要使用&&将命令链接在一起以及使用“引号”以确保读取整个命令正确地作为一个字符串。