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