在CMakeList.txt中检测32位x86处理器?

时间:2016-09-14 03:42:17

标签: cmake processor fpic

由于缺少-fPIC,我们抓住了errors in our CMake makefiles。她来自ci20 MIPS开发板:

...
[ 92%] Built target cryptopp-object
Scanning dependencies of target cryptopp-shared
Scanning dependencies of target cryptopp-static
Linking CXX static library libcryptopp.a
Linking CXX shared library libcryptopp.so
/usr/bin/ld: CMakeFiles/cryptopp-object.dir/cryptlib.cpp.o: relocation R_MIPS_HI16 against
`a local symbol' can not be used when making a shared object; recompile with -fPIC
CMakeFiles/cryptopp-object.dir/cryptlib.cpp.o: could not read symbols: Bad value
collect2: ld returned 1 exit status

由于注册压力,该项目的政策是除了32位x86以外的所有PIC。这意味着x86_64,ARM-32,Aarch32,Aarch64,MIPS,MIPS64,UltraSparc等获得PIC。

我相信目标处理器是CMAKE_SYSTEM_PROCESSOR中提供的。我遇到的问题是文档没有告诉我这些值,所以我无法弄清楚如何制作“非32位x86”测试。

如何在CMakeList.txt中检测32位x86处理器?

更好的是,我希望看到CMake设置CMAKE_SYSTEM_PROCESSOR的全面处理器列表。如果有人有这个列表,那么提供它会很棒。

2 个答案:

答案 0 :(得分:1)

我可能会在编译器周围构建一些东西。

使用现有变量/模块的近似值将是:

include(TestBigEndian)

if (NOT WIN32)
    TEST_BIG_ENDIAN(_bigendian)
    if((CMAKE_SIZEOF_VOID_P GREATER 4) OR (_bigendian))
        message(
            STATUS "Setting ${CMAKE_CXX_COMPILE_OPTIONS_PIC} "
                   "for machine ${CMAKE_HOST_SYSTEM_PROCESSOR}"
        )
        set(CMAKE_POSITION_INDEPENDENT_CODE 1)
    endif()
endif()

简而言之:

  • WIN32对64位Windows编译器/环境也有效
  • CMAKE_SIZEOF_VOID_P GREATER 4检查"大于32位"
  • 最后一个是最大的假设:将所有小端处理器作为基于Intel / AMD的
  • 使用更通用的CMAKE_POSITION_INDEPENDENT_CODE来设置-fPIC

我承认更准确的方法是围绕预定义的宏测试构建一些东西。

编辑:已添加"预定义宏检查"备选

以下是对预定义宏的更精确检查:

include(CheckCXXSourceCompiles)

if (CMAKE_CXX_COMPILE_OPTIONS_PIC)
    set(
        _preDefMacrosX86 
            __i386 __i386__ __i486__ __i586__ __i686__      
            _M_I86 _M_IX86 __X86__ _X86_ __THW_INTEL__
            __I86__ __INTEL__ __386
    )
    set(_code "void main() {}")
    foreach(_macro IN LISTS _preDefMacrosX86)
        set(
            _code
            "${_code}\n\#ifdef ${_macro}\n\#error ${_macro} is defined\n\#endif"
        )
    endforeach()
    CHECK_CXX_SOURCE_COMPILES("${_code}" _canCompileX86DoesFailCheck)

    if((CMAKE_SIZEOF_VOID_P GREATER 4) OR (_canCompileX86DoesFailCheck))
        message(STATUS "Setting ${CMAKE_CXX_COMPILE_OPTIONS_PIC}")
        set(CMAKE_POSITION_INDEPENDENT_CODE 1)
    endif()
endif()

<强>参考

答案 1 :(得分:0)

我相信这会对除Windows以外的所有内容执行检测。 Windows不会消耗-fPIC,所以对我来说无关紧要。这些碎片从三个Stack Overflow答案粘在一起。

# Stop hiding the damn output...
set(CMAKE_VERBOSE_MAKEFILE on)

# Enable PIC for all targets except Windows and 32-bit x86
if (NOT (WINDOWS OR WINDOWS_STORE OR WINDOWS_PHONE))

    set (UNAME_CMD "uname")
    set (UNAME_ARG "-m")
    execute_process(COMMAND ${UNAME_CMD} ${UNAME_ARG}
        WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}
        RESULT_VARIABLE UNAME_RESULT
        OUTPUT_VARIABLE UNAME_MACHINE)

    # Use Regex; match i386, i486, i586 and i686
    IF (NOT (${UNAME_MACHINE} MATCHES "i.86"))
        # message(STATUS "Setting -fPIC for machine ${UNAME_MACHINE}")
        if (CMAKE_VERSION VERSION_LESS 2.8.12)
            add_definitions(-fPIC)
        else()
            add_compile_options(-fPIC)
        endif()
    endif()
endif()

你得到的机器uname -m,即使在OS X上它也是最准确的。例如,在OS X上,uname -p返回i386uname -m返回{ {1}}。我似乎记得10.6或10.7有点片状,因为转换到了64位Mac。

有时你会得到x86_64的处理器,但它在很多开发板上失败了。例如,我的ci20 dev-board返回&#34; mips&#34;对于机器和&#34;未知&#34;对于处理器。另一个例子是我的LeMaker HiKey。它返回&#34; aarch64&#34;对于机器和&#34;未知&#34;对于处理器。

我仍然希望看到Cmake提供的处理器列表。