2017-05-31 85 views
0

我建立使用CMake的C库找到的符号。这里是库的CMakeLists文件:链接不能在图书馆

cmake_minimum_required(VERSION 3.7) 
project(sshserial) 

set(CMAKE_C_STANDARD 99) 

include_directories(BEFORE SYSTEM ${CMAKE_SOURCE_DIR}/common/include) 
link_directories(${CMAKE_SOURCE_DIR}/common/lib) 

set(SOURCE_FILES sshserial.c sshserial.h) 
add_library(sshserial ${SOURCE_FILES}) 

TARGET_LINK_LIBRARIES(sshserial libssh.a) 

我在C++程序中使用这个库。下面是该程序的CMakeLists文件:

cmake_minimum_required(VERSION 3.7) 
project(sshtest) 

set(CMAKE_CXX_STANDARD 11) 

include_directories(BEFORE SYSTEM ${CMAKE_SOURCE_DIR}/common/include) 
link_directories(${CMAKE_SOURCE_DIR}/common/lib) 

set(SOURCE_FILES main.cpp) 
add_executable(sshtest ${SOURCE_FILES}) 

TARGET_LINK_LIBRARIES(sshtest libsshserial.a) 

我已将该库复制到正确的文件夹中。当我在图书馆运行nm我得到这个:

$ nm libsshserial.a 

libsshserial.a(sshserial.c.o): 
       U ___error 
       U ___stack_chk_fail 
       U ___stack_chk_guard 
       U ___stderrp 
       U ___stdinp 
0000000000000330 T _create_session 
       U _exit 
       U _fgets 
       U _fprintf 
       U _free 
       U _getpass 
0000000000000490 T _read_from_session 
       U _ssh_channel_close 
       U _ssh_channel_free 
       U _ssh_channel_new 
       U _ssh_channel_open_session 
       U _ssh_channel_read 
       U _ssh_channel_request_exec 
       U _ssh_channel_send_eof 
       U _ssh_connect 
       U _ssh_disconnect 
       U _ssh_free 
       U _ssh_get_error 
       U _ssh_get_hexa 
       U _ssh_get_pubkey_hash 
       U _ssh_is_server_known 
       U _ssh_new 
       U _ssh_options_set 
       U _ssh_print_hexa 
       U _ssh_userauth_password 
       U _ssh_write_knownhost 
       U _strerror 
       U _strncasecmp 
0000000000000000 T _verify_knownhost 
       U _write 

当我尝试建立的程序,我得到这个链接错误:

Undefined symbols for architecture x86_64: 
    "create_session()", referenced from: 
     _main in main.cpp.o 
    "read_from_session(ssh_session_struct*)", referenced from: 
     _main in main.cpp.o 
ld: symbol(s) not found for architecture x86_64 
clang: error: linker command failed with exit code 1 (use -v to see invocation) 
make[3]: *** [sshtest] Error 1 
make[2]: *** [CMakeFiles/sshtest.dir/all] Error 2 
make[1]: *** [CMakeFiles/sshtest.dir/rule] Error 2 
make: *** [sshtest] Error 2 

我已经困在这里了一段时间,我我确定我错过了一些微不足道的东西。我究竟做错了什么?

P.S.我在Mac上

$ uname -a 
Darwin XXXXXX 15.6.0 Darwin Kernel Version 15.6.0: Tue Apr 11 16:00:51 PDT 2017; root:xnu-3248.60.11.5.3~1/RELEASE_X86_64 x86_64 
+0

为什么不把测试项目a *子项目*(在子文件夹中)的图书馆项目?然后你可以做'TARGET_LINK_LIBRARIES(sshtest sshserial)',CMake设置makefile以自动链接库。 –

+0

此外,当由C++编译器编译时,你会*在使用'extern“C”'声明函数时使用'extern“C”'。 –

+0

@Someprogrammerdude:我现在可以尝试,我会的。但后来我想在其他项目中使用C库,所以它不是我想的永久解决方案。 – nakiya

回答

2

问题是C++使用name mangling。要禁止这种情况,您需要用extern "C"声明C函数,但仅限于由C++编译器编译时。你可以编译C.

当在声明函数原型的头文件与__cplusplus宏,这是编译C++时定义的查询,但在这,这样做

#ifdef __cplusplus 
// Being compiled by a C++ compiler, inhibit name mangling 
extern "C" { 
#endif 

// All your function prototype declarations here 

#ifdef __cplusplus 
} // End of extern "C" 
#endif 
+0

正如我在文章上的评论指出,该库以C,所以我不需要用'为extern“C”来装点导出符号' – nakiya

+3

@nakiya如果你想使用C++的功能,那么您***需要它!故事结局!去尝试一下。 –

+0

好的,我明白你的观点。我会试试这个。 – nakiya