Android NDK 上“__aeabi_ul2f”的多重定义(libgcc_real.a)

问题描述 投票:0回答:2

我在使用 CMake 在 Android 上编译大型项目时收到有关

multiple definition of '__aeabi_ul2f'
的错误。显然,找到定义的地方之一是在

libsmoltcp_cpp_interface_rust.a(compiler_builtins-5ae026d08032e786.compiler_builtins.2kv22hwk-cgu.127.rcgu.o): previous definition here

这是一些与 C++ 混合的 Rust 代码,在 CMake 中像这样编译:

add_custom_target(
    lib_smol_tcp_rust
    COMMAND 
    echo "-------------COMPILING LIBSMOLTCP_CPP_INTERFACE RELEASE x86_64"
    && cargo build --release
    && echo "-------------COMPILING LIBSMOLTCP_CPP_INTERFACE RELEASE aarch64-linux-android"
    && cargo build --target aarch64-linux-android --release 
    && echo "-------------COMPILING LIBSMOLTCP_CPP_INTERFACE RELEASE armv7-linux-androideabi"
    && cargo build --target armv7-linux-androideabi --release 
    && echo "-------------COMPILING LIBSMOLTCP_CPP_INTERFACE RELEASE i686"
    && cargo build --target i686-linux-android --release
    WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
)

然后

lsmoltcp_cpp_interface_rust
(C++ 代码)链接到
lib_smol_tcp_rust
(Rust)。

看起来对象

'__aeabi_ul2f'
已经在 Android NDK 项目的
libgcc_real.a
上定义了。

所以我不应该针对 libgcc 编译 C++/Rust 混合代码?我怎样才能做到这一点?我很困惑,因为我使用

整个错误:

  [1369/1369] Linking CXX shared library /home/dev/orwell/orwell_flutter_app/build/app/intermediates/cmake/debug/obj/armeabi-v7a/liborwell_android.so
  FAILED: /home/dev/orwell/orwell_flutter_app/build/app/intermediates/cmake/debug/obj/armeabi-v7a/liborwell_android.so 
  : && /opt/android-sdk-linux/ndk/21.1.6352462/toolchains/llvm/prebuilt/linux-x86_64/bin/clang++ --target=armv7-none-linux-androideabi24 --gcc-toolchain=/opt/android-sdk-linux/ndk/21.1.6352462/toolchains/llvm/prebuilt/linux-x86_64 --sysroot=/opt/android-sdk-linux/ndk/21.1.6352462/toolchains/llvm/prebuilt/linux-x86_64/sysroot -fPIC -g -DANDROID -fdata-sections -ffunction-sections -funwind-tables -fstack-protector-strong -no-canonical-prefixes -D_FORTIFY_SOURCE=2 -march=armv7-a -mthumb -Wformat -Werror=format-security  -std=c++17 -O0 -fno-limit-debug-info  -Wl,--exclude-libs,libgcc.a -Wl,--exclude-libs,libgcc_real.a -Wl,--exclude-libs,libatomic.a -static-libstdc++ -Wl,--build-id -Wl,--fatal-warnings -Wl,--exclude-libs,libunwind.a -Wl,--no-undefined -Qunused-arguments -shared -Wl,-soname,liborwell_android.so -o /home/dev/orwell/orwell_flutter_app/build/app/intermediates/cmake/debug/obj/armeabi-v7a/liborwell_android.so CMakeFiles/orwell_android.dir/orwell_jni.cpp.o CMakeFiles/orwell_android.dir/DecodedFfmpegFrameJNI.cpp.o CMakeFiles/orwell_android.dir/JavaOrwellFlutterRenderer.cpp.o CMakeFiles/orwell_android.dir/MediaCodecDecoder.cpp.o CMakeFiles/orwell_android.dir/JavaSimpleFileWriter.cpp.o CMakeFiles/orwell_android.dir/JavaHashMapJNI.cpp.o CMakeFiles/orwell_android.dir/JavaFlutterEventMessenger.cpp.o  _liborwell/liborwell_static.a -landroid -llog -lmediandk /home/dev/orwell/orwell_flutter_app/android/app/src/main/cpp/../../../../../../deps/ffmpeg/build/android/armeabi-v7a/lib/libavcodec.so /home/dev/orwell/orwell_flutter_app/android/app/src/main/cpp/../../../../../../deps/ffmpeg/build/android/armeabi-v7a/lib/libavutil.so /home/dev/orwell/orwell_flutter_app/android/app/src/main/cpp/../../../../../../deps/ffmpeg/build/android/armeabi-v7a/lib/libswscale.so /home/dev/orwell/orwell_flutter_app/android/app/src/main/cpp/../../../../../../deps/ffmpeg/build/android/armeabi-v7a/lib/libswresample.so /home/dev/orwell/orwell_flutter_app/build/app/intermediates/cmake/debug/obj/armeabi-v7a/libmyRtspClient.so /home/dev/orwell/orwell_flutter_app/build/app/intermediates/cmake/debug/obj/armeabi-v7a/libjrtp.so _liborwell/_myRtspClient/libmyRtspClient-static.a _liborwell/_myRtspClient/_JTRPLIB/src/libjrtp.a _liborwell/_ZLMediaKit/libzlmediakit.a _liborwell/_ZLMediaKit/libzltoolkit.a _liborwell/_ZLMediaKit/libmpeg.a _liborwell/_ZLMediaKit/libmov.a _liborwell/_ZLMediaKit/libflv.a _liborwell/common/openvpn_zl_socket/libopenvpn_zl_socket.a _liborwell/_libopenvpn3/src/libopenvpn/libopenvpn_lib.a _liborwell/_libopenvpn3/openvpn3/libssl.a _liborwell/_libopenvpn3/openvpn3/libcrypto.a _liborwell/_libopenvpn3/openvpn3/liblzo.a _liborwell/_libopenvpn3/openvpn3/liblz4.a _liborwell/_libopenvpn3/libtins/lib/libtins.a _liborwell/_libopenvpn3/smoltcp_cpp_interface/libsmoltcp_cpp_static.a /home/dev/orwell/deps/libopenvpn3/smoltcp_cpp_interface/target/armv7-linux-androideabi/release/libsmoltcp_cpp_interface_rust.a -ldl _liborwell/common/liborwellebml/liborwell_ebml.a _liborwell/_libebml/libebml.a _liborwell/common/liborwellprofile/liborwell_profile.a _liborwell/common/liborwellprofile2/liborwell_profile2.a _liborwell/common/liborwellprofile2/_protobuf/libprotobufd.a /opt/android-sdk-linux/ndk/21.1.6352462/toolchains/llvm/prebuilt/linux-x86_64/sysroot/usr/lib/arm-linux-androideabi/libz.a -latomic -lm && :
  /opt/android-sdk-linux/ndk/21.1.6352462/toolchains/llvm/prebuilt/linux-x86_64/lib/gcc/arm-linux-androideabi/4.9.x/../../../../arm-linux-androideabi/bin/ld: error: /opt/android-sdk-linux/ndk/21.1.6352462/toolchains/llvm/prebuilt/linux-x86_64/lib/gcc/arm-linux-androideabi/4.9.x/armv7-a/thumb/libgcc_real.a(_arm_addsubsf3.o): multiple definition of '__aeabi_ul2f'
  /opt/android-sdk-linux/ndk/21.1.6352462/toolchains/llvm/prebuilt/linux-x86_64/lib/gcc/arm-linux-androideabi/4.9.x/../../../../arm-linux-androideabi/bin/ld: /home/dev/orwell/deps/libopenvpn3/smoltcp_cpp_interface/target/armv7-linux-androideabi/release/libsmoltcp_cpp_interface_rust.a(compiler_builtins-5ae026d08032e786.compiler_builtins.2kv22hwk-cgu.127.rcgu.o): previous definition here
  clang++: error: linker command failed with exit code 1 (use -v to see invocation)                                
  ninja: build stopped: subcommand failed.    

                                         

这就是我创建 Rust 工具链的方式:

&& ${ANDROID_NDK_HOME}/build/tools/make_standalone_toolchain.py --api 26 --arch arm64 --install-dir /opt/RUST_NDK_TOOLCHAIN/arm64 \
    && ${ANDROID_NDK_HOME}/build/tools/make_standalone_toolchain.py --api 26 --arch arm --install-dir /opt/RUST_NDK_TOOLCHAIN/arm \
    && ${ANDROID_NDK_HOME}/build/tools/make_standalone_toolchain.py --api 26 --arch x86 --install-dir /opt/RUST_NDK_TOOLCHAIN/x86

如您所见,链接使用

--exclude-libs,libgcc_real.a

所以它根本不应该使用

libgcc_real.a
,但它无论如何都会抱怨
libgcc_real.a

android c++ rust android-ndk linker
2个回答
1
投票

在 C/C++ 项目中链接 Rust staticlib 之前我遇到过这个问题。

让 gcc 不链接 libgcc 是非常困难的(有时是不可能的)。它被硬编码为“默认”库,链接器将始终拉取它(您可以尝试

-nodefaultlibs
-nostdlib
来防止这种情况)。它也可以从链接描述文件中拉入。

对我来说,窍门是链接器参数中的顺序很重要。

链接器的工作原理(大致)如下:

  • 所有参数均从左到右处理,每个参数仅处理一次。
  • 对于每个非
    -l
    参数,链接器加载给定的.a/.o并拉入所有符号,如果有重复则出错。如果拉入的符号需要其他未在任何地方定义的符号,链接器会将它们作为“未定义的符号”进行跟踪。
  • 对于每个
    -l
    参数,链接器加载 .a 并仅拉入与之前未定义的符号匹配的符号。如果 .a 具有已定义的符号,它们将被忽略(而不是出错!)。

尝试尽早将

libsmoltcp_cpp_interface_rust.a
放入链接器参数中。任何位于其后面的
-l
都不会导致重复错误,因为
-l
会忽略重复项。


0
投票

我的错误,导致这个问题的是:

我手动添加了类似的内容 {$R *.dres} {$R myapp.dres}

并且还拥有 {$R myapp.dres} 在一个单位

因此资源是多重定义的。

正确且足够的是: {$R *.dres} 在项目源中

© www.soinside.com 2019 - 2024. All rights reserved.