【问题标题】:How to set up CMake to cross compile with clang for ARM embedded on Windows?如何设置 CMake 以使用 clang 为 Windows 上嵌入的 ARM 进行交叉编译?
【发布时间】:2019-06-29 13:59:46
【问题描述】:

我正在尝试生成 Ninja makefile 以使用 Clang 为 ARM Cortex A5 CPU 交叉编译 C++ 项目。我为 CMake 创建了一个工具链文件,但似乎有一个错误或我找不到的东西丢失。当使用下面的工具链文件调用 CMake 时,出现以下错误。

CMake 命令:

cmake -DCMAKE_TOOLCHAIN_FILE="..\Src\Build\Toolchain-clang-arm.cmake" -GNinja ..\Src\

输出:

-- C 编译器标识为 Clang 7.0.0 CMake Error at C:/Users/user/scoop/apps/cmake/3.13.4/share/cmake-3.13/Modules/CMakeDetermineCompilerId.cmake:802 (消息):Clang 编译器工具

“C:/Program Files/LLVM/bin/clang.exe”

以 MSVC ABI 为目标,但具有类似 GNU 的命令行界面。 这是不支持的。请改用“clang-cl”,例如通过设置 'CC=clang-cl' 在环境中。此外,使用 MSVC 命令行环境。调用堆栈(最近的调用优先):
C:/Users/user/scoop/apps/cmake/3.13.4/share/cmake-3.13/Modules/CMakeDetermineCCompiler.cmake:113 (CMAKE_DIAGNOSE_UNSUPPORTED_CLANG) CMakeLists.txt:2 (项目)

-- 配置不完整,出现错误!

CMake 工具链文件(Toolchain-clang-arm.cmake):

set(CMAKE_CROSSCOMPILING TRUE)
SET(CMAKE_SYSTEM_NAME Generic)
set(CMAKE_SYSTEM_PROCESSOR arm)

# Clang target triple
SET(TARGET armv7-none-eabi)

# specify the cross compiler
SET(CMAKE_C_COMPILER_TARGET ${TARGET})
SET(CMAKE_C_COMPILER clang)
SET(CMAKE_CXX_COMPILER_TARGET ${TARGET})
SET(CMAKE_CXX_COMPILER clang++)
SET(CMAKE_ASM_COMPILER_TARGET ${TARGET})
SET(CMAKE_ASM_COMPILER clang)

# C/C++ toolchain
SET(TOOLCHAIN "C:/Program Files (x86)/GNU Tools ARM Embedded/7 2018-q2-update")
SET(CMAKE_C_COMPILER_EXTERNAL_TOOLCHAIN ${TOOLCHAIN})
SET(CMAKE_CXX_COMPILER_EXTERNAL_TOOLCHAIN ${TOOLCHAIN})

# specify compiler flags
SET(ARCH_FLAGS "-target armv7-none-eabi -mcpu=cortex-a5")
SET(CMAKE_C_FLAGS "-Wall -Wextra ${ARCH_FLAGS}" CACHE STRING "Common flags for C compiler")
SET(CMAKE_CXX_FLAGS "-Wall -Wextra -std=c++11 -fno-exceptions -fno-threadsafe-statics ${ARCH_FLAGS}" CACHE STRING "Common flags for C++ compiler")

我使用CMakeClang 文档以及来自网络的一些random link 创建工具链文件。整个项目使用 ARM GCC for Windows 编译良好,因此工具链文件似乎是唯一缺失的部分。

编辑

我试图通过强制编译器来解决 CMake 编译器检查问题。我将这些行替换为 SET(CMAKE_C_COMPILER clang)、SET(CMAKE_CXX_COMPILER clang++) 等:

CMAKE_FORCE_C_COMPILER(clang Clang)
CMAKE_FORCE_CXX_COMPILER(clang++ Clang)

错误保持不变。

编辑

我可以使用 clang -target arm-none-eabi 成功编译一个 hello world 示例。所以问题似乎出在 CMake 中。我在 CMake 问题跟踪器中创建了一个bug

工具版本:

  • clang 版本 7.0.0 (tags/RELEASE_700/final)
  • cmake 版本 3.13.4

【问题讨论】:

  • 您在 CMake GitLab 问题上留下了评论,表明您可能已经成功使用 msys2 和 clang 包。如果是这样,您能否在此处发布答案/评论,详细说明您如何使其工作?我仍然没有成功,我相信我正在尝试做同样的事情。
  • @AngCaruso 感谢您提醒写一个答案。我希望这能回答你的问题。

标签: c++ windows cmake arm clang


【解决方案1】:

我的问题已通过我的错误报告中的reply 得到解答,但我在此处添加了答案,以便将所有信息集中在一个位置以供将来参考。

简而言之:如果您从llvm.org 安装 Clang,CMake 目前不支持使用 clang/clang++ 命令行界面。如果您想使用 clang/clang++ 接口(对于 ARM 进行交叉编译是必需的),您必须通过 msys2 安装 Clang。

详细说明

Windows 上的 Clang 有两种不同的命令行界面:

  • clang/clang++ 尝试与 GCC gcc/g++ 兼容并针对 GNU ABI 的默认接口
  • clang-cl 试图与 Microsoft 的 Visual C++ 编译器 cl.exe 兼容并针对 MSVC ABI

要为 ARM 进行交叉编译,您需要 clang/clang++ 接口。问题是 CMake 支持不同的接口,具体取决于您安装 Clang 的方式(有关更多详细信息,请参阅 CMake 问题跟踪器中的bug):

  • 如果您从llvm.org 安装 Clang,CMake 仅支持 clang-cl 接口。
  • 如果您通过msys2 安装 Clang,CMake 支持 clang/clang++ 接口。

这就是我所做的:

  1. 安装msys2
  2. 使用pacman 安装 Clang 和 CMake。 msys2中有两个clang包,一个mingw32和一个mingw64版本。我使用了 mingw64 包 (mingw-w64-x86_64-clang)。
  3. 启动 mingw64 shell 并运行 CMake 并从那里构建。

工具链文件

我的原始工具链文件有两个问题需要我很长时间才能解决。所以我希望这可以节省其他人一些时间:

  1. target triple(例如 arm-none-eabi)需要与 GCC binutils 的前缀完全匹配。我的 binutils 的前缀是 arm-none-eabi(例如 arm-none-eabi-ar),所以我必须相应地更改目标三元组。
  2. CMAKE_TRY_COMPILE_TARGET_TYPE 需要更改为 STATIC_LIBRARY 以防止 CMake 在编译检查期间运行链接器。

这是我使用的最终工具链文件(您也可以在 GitHub repo 中找到一个很好的工具链文件示例):

cmake_minimum_required(VERSION 3.13)

set(CMAKE_SYSTEM_NAME Generic)
set(CMAKE_SYSTEM_PROCESSOR ARM)

if(DEFINED ENV{GCC_ARM_TOOLCHAIN})
    set(GCC_ARM_TOOLCHAIN $ENV{GCC_ARM_TOOLCHAIN})
else()
    set(GCC_ARM_TOOLCHAIN "C:/Users/user/tools/gcc-arm-none-eabi-7-2018-q2-update-win32")
endif()

LIST(APPEND CMAKE_PROGRAM_PATH ${GCC_ARM_TOOLCHAIN})

# Specify the cross compiler
# The target triple needs to match the prefix of the binutils exactly
# (e.g. CMake looks for arm-none-eabi-ar)
set(CLANG_TARGET_TRIPLE arm-none-eabi)
set(GCC_ARM_TOOLCHAIN_PREFIX ${CLANG_CLANG_TARGET_TRIPLE})
set(CMAKE_C_COMPILER clang)
set(CMAKE_C_COMPILER_TARGET ${CLANG_TARGET_TRIPLE})
set(CMAKE_CXX_COMPILER clang++)
set(CMAKE_CXX_COMPILER_TARGET ${CLANG_TARGET_TRIPLE})
set(CMAKE_ASM_COMPILER clang)
set(CMAKE_ASM_COMPILER_TARGET ${CLANG_TARGET_TRIPLE})

# Don't run the linker on compiler check
set(CMAKE_TRY_COMPILE_TARGET_TYPE STATIC_LIBRARY)

# Specify compiler flags
set(ARCH_FLAGS "-mcpu=cortex-a5 -mthumb -mfpu=neon-vfpv4 -mfloat-abi=hard -mno-unaligned-access")
set(CMAKE_C_FLAGS "-Wall ${ARCH_FLAGS}" CACHE STRING "Common flags for C compiler")
set(CMAKE_CXX_FLAGS "-Wall -std=c++17 -fno-exceptions -fno-rtti -fno-threadsafe-statics ${ARCH_FLAGS}" CACHE STRING "Common flags for C++ compiler")
set(CMAKE_ASM_FLAGS "-Wall ${ARCH_FLAGS} -x assembler-with-cpp" CACHE STRING "Common flags for assembler")
set(CMAKE_EXE_LINKER_FLAGS "-nostartfiles -Wl,-Map,kernel.map,--gc-sections -fuse-linker-plugin -Wl,--use-blx --specs=nano.specs --specs=nosys.specs" CACHE STRING "")

# C/C++ toolchain
set(GCC_ARM_SYSROOT "${GCC_ARM_TOOLCHAIN}/${GCC_ARM_TOOLCHAIN_PREFIX}")
# set(CMAKE_SYSROOT ${GCC_ARM_SYSROOT})
set(CMAKE_FIND_ROOT_PATH ${GCC_ARM_SYSROOT})

# Search for programs in the build host directories
set(CMAKE_FIND_ROOT_PATH_MODE_PROGRAM NEVER)
# For libraries and headers in the target directories
set(CMAKE_FIND_ROOT_PATH_MODE_LIBRARY ONLY)
set(CMAKE_FIND_ROOT_PATH_MODE_INCLUDE ONLY)

【讨论】:

  • 感谢您发布此信息。我最初的问题实际上是使用 GGC for ARM 工具链从这里 gnu-mcu-eclipse.github.io/toolchain/arm/install 进行交叉编译,此后我已经想出了如何成功地做到这一点。我可能会在这里发布一个答案,即使它没有回答你的具体问题。我也试过在msys2中安装cmake,但奇怪的是不支持MinGW生成器的MSYS,这意味着我必须继续使用单独安装的原生Windows cmake。
【解决方案2】:

这是一个替代答案,它使用 CMake 使用 GCC 工具链而不是 clang 为嵌入式 ARM 进行交叉编译,并且还使用普通的旧 make 而不是 Ninja。我知道这并不能直接回答问题,但它是嵌入式 ARM 的一个非常合理的替代方案,而且我们现在使用的许多基于 Eclipse 的供应商工具链都是基于 GCC 而不是 LLVM/clang。

首先从https://github.com/gnu-mcu-eclipse/arm-none-eabi-gcc/releases 安装最新的“GNU MCU Eclipse ARM Embedded GCC”工具链(注意:无需安装程序,只需解压缩到您想要的位置即可)。

接下来,使用https://www.msys2.org/ 提供的安装程序安装最新的 MSYS2/MinGW。

由于某种原因,MSYS2 的默认安装不包括 make。因此,在按照安装说明使用 pacman 更新默认软件包后,执行 'pacman -S make' 来安装 make。

在 MSYS2 中有一个可用的 cmake 构建。但是,它不包括 MSYS 或 MinGW 的生成器(不知道为什么),并且(或因为)似乎在查找外部 GCC 工具链时遇到问题,即使在命令行或工具链文件中明确提供了完整路径.因此,此解决方案使用 CMake 的本机 Window 构建,可在 https://cmake.org/download/ 获得,而不是 cmake 的 MSYS2 构建。

默认情况下,MSYS shell 不继承 Windows 环境,因此本机 Windows CMake 不会位于 MSYS shell 路径上。简单的解决方案是在命令行上指定 cmake 的完整路径。就我而言,我在 C:/GNU MCU Eclipse/ARM Embedded GCC/8.2.1-1.4-20190214-0604 有 GCC 交叉编译器工具链。因此,一个完整的 cmake 命令如下所示。

/c/Program\ Files/CMake/bin/cmake.exe -DCMAKE_BUILD_TYPE=Debug -DCMAKE_TOOLCHAIN_FILE=cmake/toolchain-arm-none-eabi.cmake "-DTOOLCHAIN_PREFIX=C:/GNU MCU Eclipse/ARM Embedded GCC/8.2.1-1.4-20190214-0604" -G "MSYS Makefiles" ../

这里的命令是从项目根目录下一级的构建目录执行的,同级 cmake 目录包含指定的工具链文件。请注意,反斜杠对于转义 cmake.exe 路径中的任何空格是必需的,但在作为参数传递给 cmake 的引用路径中不使用。

只要 cmake 和 msys bin 目录都在 Windows 路径上,您也可以直接从普通的旧 Windows 命令行 (cmd.exe) 运行 cmake 和 make。在这种情况下,无需指定 cmake 的完整路径。然而,PowerShell 没有成功。

【讨论】:

  • 你是对的,如果你想为嵌入式 ARM 设备建立一个工具链,使用 GCC 是更合理的选择。事实上,这就是现在这个项目中使用的东西。我希望能够使用 Clang 编译项目的主要原因是使用 Clang 提供的出色工具,例如 clang-formatclang-tidy。对于部署在目标上的二进制文件,我们仍将使用 GCC。
猜你喜欢
  • 2017-10-02
  • 2014-05-31
  • 2023-04-02
  • 2012-10-02
  • 2013-01-19
  • 1970-01-01
  • 1970-01-01
  • 2015-10-21
  • 1970-01-01
相关资源
最近更新 更多