所以我为我的问题设计了一个相当有创意的解决方案……看来 CMake 没有任何功能来检测目标架构。
现在,我们知道我们可以在 C 中轻松做到这一点,因为像__i386__
,__x86_64__
等符号将根据您的环境进行定义。幸运的是,CMake 有一个 try_run 函数,它将在配置阶段编译和运行任意 C 源代码文件。
然后我们可以编写一个小程序,它使用一堆 ifdef 并将架构名称作为字符串写入控制台。唯一的问题是,这仅在主机和目标系统相同时才有效……它在交叉编译期间无法工作,因为虽然您可以编译二进制文件,但无法运行它来查看其输出。
这就是事情变得有趣的地方。我们可以通过故意编写一个损坏的 C 程序来利用 C 预处理器来获取必要的信息……我们使用基于 ifdefs 将架构名称写入控制台的原始概念,但我们不会这样做,而是简单地放置一个 #错误预处理器指令代替 printf 调用。
当 CMake 的 try_run 函数编译 C 文件时,编译总是会失败,但是我们在 #error 指令中放置的任何消息都会显示在编译器的错误输出中,try_run 返回给我们。
因此,我们所要做的就是使用一些 CMake 字符串命令从编译器的错误输出中解析架构名称,并且我们可以检索目标架构......即使在交叉编译时也是如此。
代码的 OS X 特定部分主要使用 CMAKE_OSX_ARCHITECTURES 来确定目标架构,但在未指定的情况下,它将使用与其他系统相同的代码并正确返回 x86_64(对于现代系统,这是编译器的默认值)或 i386 (对于较旧的 OS X 系统,例如 Leopard)。
我已经使用 Visual Studio 9 和 10 生成器(x86、x86_64、ia64)、Xcode、NMake、MSYS Makefile 和 Unix Makefile 在 Windows、OS X 和 Linux 上测试并验证了它的工作原理。每次都返回正确的结果。
注意:如果您故意将 -m32 或 -m64 传递给编译器或其他可能影响目标体系结构的标志(有没有办法将所有环境设置传递给 try_run?),此解决方案可能会失败;这不是我测试过的东西。只要您使用生成器的默认设置并且所有目标都针对相同的体系结构进行编译,您应该没问题。
我的解决方案的完整源代码可以在 GitHub 上找到:https ://github.com/petroules/solar-cmake/blob/master/TargetArch.cmake