由于未定义对具有 std::string 的方法的引用,链接 webrtc-native 时出错

Posted

技术标签:

【中文标题】由于未定义对具有 std::string 的方法的引用,链接 webrtc-native 时出错【英文标题】:Error linking webrtc-native due to undefined reference to methods having std::string 【发布时间】:2017-11-17 10:20:14 【问题描述】:

我正在尝试构建 webrtc 版本 62,使用以下内容

1.git checkout -b branch62 refs/remotes/branch-heads/62
2.gn gen out_release_62/x64/Debug --args="rtc_include_tests=false rtc_use_h264=false use_rtti=true is_component_build=false enable_iterator_debugging=false enable_nacl=false target_os=\"linux\" target_cpu=\"x64\" is_debug=true"
3.ninja -C out_release_62/x64/Debug

我正在链接到 libwebrtc.a 和 libwebrtc_common.a

但无论我尝试使用 RTC_CHECK_EQ 或 RTC_DCHECK_LT,我都会收到以下错误

undefined reference to `rtc::FatalMessage::FatalMessage(char const*, int, std::string*)'

out_release_62/include/webrtc/rtc_base/checks.h:176: undefined reference to `std::string* rtc::MakeCheckOpString<int, int>(int const&, int const&, char const*)'
./src/testwebrtc.o: In function `rtc::CheckLtImpl(int, int, char const*)':

我对 build branch head 60 没有任何问题, 我尝试链接到 librtc_base.alibrtc_base_approved.a 但没有运气。 产生此错误的示例程序是

#include <iostream>
#include <string>
#include <webrtc/rtc_base/checks.h>
#include <webrtc/rtc_base/ssladapter.h>
#include <webrtc/rtc_base/logging.h>
#include <webrtc/api/peerconnectioninterface.h>
using namespace std;

int main() 
    cout << "!!!Hello World!!!" << endl; // prints !!!Hello World!!!
    std::string type;
    std::string sdp;
    webrtc::SdpParseError error;
    webrtc::CreateSessionDescription(type,sdp,&error);
    RTC_DCHECK(5);
    RTC_CHECK_EQ(1,6);
    RTC_DCHECK_LT(1,4);
    return 0;

gcc 版本 6.3 链接器

g++ -L/home/test/webrtc-checkout-ankur/src/out_release_62/x64/Debug -o "testwebrtc"  ./src/testwebrtc.o   -lwebrtc -lwebrtc_common -lpthread -lm -ldl

这是返回的错误

./src/testwebrtc.o: In function `main':
/home/test/workspace/testwebrtc/Debug/../src/testwebrtc.cpp:22: undefined reference to `webrtc::CreateSessionDescription(std::string const&, std::string const&, webrtc::SdpParseError*)'
/home/test/workspace/testwebrtc/Debug/../src/testwebrtc.cpp:24: undefined reference to `rtc::FatalMessage::FatalMessage(char const*, int, std::string*)'
/home/test/workspace/testwebrtc/Debug/../src/testwebrtc.cpp:25: undefined reference to `rtc::FatalMessage::FatalMessage(char const*, int, std::string*)'
./src/testwebrtc.o: In function `rtc::CheckEqImpl(int, int, char const*)':
makefile:45: recipe for target 'testwebrtc' failed
/home/test/webrtc-checkout-ankur/src/out_release_62/include/webrtc/rtc_base/checks.h:176: undefined reference to `std::string* rtc::MakeCheckOpString<int, int>(int const&, int const&, char const*)'
./src/testwebrtc.o: In function `rtc::CheckLtImpl(int, int, char const*)':
/home/test/webrtc-checkout-ankur/src/out_release_62/include/webrtc/rtc_base/checks.h:179: undefined reference to `std::string* rtc::MakeCheckOpString<int, int>(int const&, int const&, char const*)'
collect2: error: ld returned 1 exit status

根据我的观察,任何包含 std::string 的函数/方法都显示出这种行为。 _GLIBCXX_USE_CXX11_ABI=0 已设置,并且 -std=c++0x ,此构建是在 ubuntu 16.04 系统上完成的。

在我得到的构建库上运行 nm

find -iname '.a' -exec nm -A -C \; | grep 'T rtc::FatalMessage::FatalMessage('

./Debug/obj/webrtc/rtc_base/librtc_base_approved.a:checks.o:00000000000001a0 T rtc::FatalMessage::FatalMessage(char const*, int)
./Debug/obj/webrtc/rtc_base/librtc_base_approved.a:checks.o:00000000000005e0 T rtc::FatalMessage::FatalMessage(char const*, int, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> >*)
./Debug/obj/webrtc/rtc_base/librtc_base_approved.a:checks.o:00000000000001a0 T rtc::FatalMessage::FatalMessage(char const*, int)
./Debug/obj/webrtc/rtc_base/librtc_base_approved.a:checks.o:00000000000005e0 T rtc::FatalMessage::FatalMessage(char const*, int, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> >*)
./Debug/obj/webrtc/libwebrtc.a:checks.o:00000000000001a0 T rtc::FatalMessage::FatalMessage(char const*, int)
./Debug/obj/webrtc/libwebrtc.a:checks.o:00000000000005e0 T rtc::FatalMessage::FatalMessage(char const*, int, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> >*)
./Debug/obj/webrtc/libwebrtc.a:checks.o:00000000000001a0 T rtc::FatalMessage::FatalMessage(char const*, int)
./Debug/obj/webrtc/libwebrtc.a:checks.o:00000000000005e0 T rtc::FatalMessage::FatalMessage(char const*, int, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> >*)

find -iname '.a' -exec nm -A -C \; | grep 'CreateSessionDescription('

./Debug/obj/webrtc/pc/libpeerconnection.a:jsepsessiondescription.o:0000000000000000 T webrtc::CreateSessionDescription(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&, webrtc::SdpParseError*)
./Debug/obj/webrtc/libwebrtc.a:jsepsessiondescription.o:0000000000000000 T webrtc::CreateSessionDescription(std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&, std::__1::basic_string<char, std::__1::char_traits<char>, std::__1::allocator<char> > const&, webrtc::SdpParseError*)

请帮忙

【问题讨论】:

试试-l/path/to/lib/libwebrtc.a 我已经在做。所有在签名链接中没有 std::string 的方法都正确 【参考方案1】:

你有一个标准的库不匹配:一方面你使用 std::string,另一方面你使用 std::__1::basic_string

换句话说:一方面你使用 libc++,另一方面你使用 libstdc++

你需要使用clang并通过

-stdlib=libstdc++

到编译器命令行。

__1 表示我们正在使用 libc++ llvm-clang 库,这是 webrtc 构建的默认库

__1 是 libc++ 使用的内联命名空间,

为了解决这个特殊问题,当我们链接到 libwebrtc 时,我们应该在我们的应用程序中使用 libc++ 而不是 libstdc++,

构建 webrtc 以使用 libstdc++, 为了让 gn 知道我们打算使用 libstdc++,我们将 use_custom_libcxx=false 和 use_custom_libcxx_for_host=false 作为 args 传递

【讨论】:

感谢您的解释,这让我畅通无阻!顺便说一句,我认为即使使用 GCC 编译 libc++,当 gn 没有提供您提到的参数时也是如此。

以上是关于由于未定义对具有 std::string 的方法的引用,链接 webrtc-native 时出错的主要内容,如果未能解决你的问题,请参考以下文章

在 C++ 中比较两个 std::string 时的未定义行为 [关闭]

与静态库中的 std::string 相关的 C++ 未定义符号

使用 fmt 链接错误:未定义对 `std::string fmt::v6::internal::grouping_impl<char>(fmt::v6::internal::locale

未定义对 YAML::LoadFile 的引用

未定义对 cv::fontQt 的引用

带有自定义分配器的 std::string