链接英特尔MKL静态库引入了循环依赖。当我导入库时,
set(LIBRARIES mkl_intel_lp64 mkl_sequential mkl_core)
foreach(_lib ${LIBRARIES})
add_library(${_lib} UNKNOWN IMPORTED)
set_target_properties(${_lib} PROPERTIES IMPORTED_LOCATION
/opt/intel/mkl/lib/intel64/lib${_lib}.a)
endforeach()
并链接到我的可执行文件
target_link_libraries(main PRIVATE ${LIBRARIES})
我得到了大量对线性代数调用的未定义引用。例如,
ztrevc3_gen.f:(.text+0x1af7): undefined reference to `mkl_blas_zdscal'
解决此问题的一种方法是使用适当的链接器标志:
target_link_libraries(main PRIVATE -Wl,--start-group ${LIBRARIES} -Wl,--end-group)
另一种选择是这样做:
target_link_libraries(main PRIVATE ${LIBRARIES} ${LIBRARIES} ${LIBRARIES})
然而,当我在寻找更优雅的解决方案时,我遇到了LINK_INTERFACE_MULTIPLICITY
属性。如果将此属性与导入的库位置一起设置,
set(LIBRARIES mkl_intel_lp64 mkl_sequential mkl_core)
foreach(_lib ${LIBRARIES})
add_library(${_lib} UNKNOWN IMPORTED)
set_target_properties(${_lib} PROPERTIES IMPORTED_LOCATION
/opt/intel/mkl/lib/intel64/lib${_lib}.a
LINK_INTERFACE_MULTIPLICITY 3)
endforeach()
我得到了和以前一样的未定义引用,所以显然这不起作用。使用LINK_INTERFACE_MULTIPLICITY的正确方法是什么,是否有一种更优雅的方式来绕过循环依赖?
修改
这是一个失败的最小示例,这次是正确的IMPORTED_LINK_INTERFACE_MULTIPLICITY
变量。
# CMakeLists.txt
cmake_minimum_required(VERSION 3.10)
project(test Fortran)
add_executable(main main.f90)
set(LIBRARIES mkl_intel_lp64 mkl_sequential mkl_core)
foreach(_lib ${LIBRARIES})
add_library(${_lib} UNKNOWN IMPORTED)
set_target_properties(${_lib} PROPERTIES IMPORTED_LOCATION
/opt/intel/mkl/lib/intel64/lib${_lib}.a
IMPORTED_LINK_INTERFACE_MULTIPLICITY 3)
endforeach()
list(APPEND LIBRARIES dl pthread)
target_link_libraries(main ${LIBRARIES})
#target_link_libraries(main ${LIBRARIES} ${LIBRARIES} ${LIBRARIES})
# main.f90
call zpotrf
end program
如果取消注释最后一行,则构建成功。不幸的是,MKL并不是免费的(在某些情况下除外),但希望有人可以测试一下。我应该注意到它失败了一些线性代数调用而不是其他类似dgemm
。
答案 0 :(得分:0)
属性LINK_INTERFACE_MULTIPLICITY用于"正常" STATIC 库。对于 IMPORTED 库,应使用IMPORTED_LINK_INTERFACE_MULTIPLICITY属性:
set_target_properties(${_lib} PROPERTIES
IMPORTED_LOCATION /opt/intel/mkl/lib/intel64/lib${_lib}.a
IMPORTED_LINK_INTERFACE_MULTIPLICITY 3)