Android C++ 异常抛出导致 SIGSEGV

标签 android c++ cmake android-ndk segmentation-fault

我在将 C++ 代码集成到我的 Android 应用程序时遇到了一些问题。 我使用 Djinni 库生成我在应用程序中使用的 JNI 代码。 我正在使用 CMake 构建我的 C++ 库,并在此过程中链接 OpenCV 库(带有 .so 文件)。 我使用 NDK r19。

我有一个集成到项目中的 C++ 类,它可以抛出一个异常,一个 std::runtime_error,但是这个异常有一个我不理解的行为。

第一个问题:

我在这里查看了文档 https://developer.android.com/ndk/guides/cmake并使用属性 -DANDROID_CPP_FEATURES=rtti exceptions 启用 C++ 异常。 这样,一旦在 C++ 中抛出我的异常,我就会收到一个 SIGABRT,这会使我的应用程序崩溃。 即使 C++ 中的 throw 被 catch(...) block 包围,也会发生这种情况。 我还尝试通过 cppFlags 应用标志 -fexceptions-frtti 但没有成功。

第二个问题:

我使用了另一个具有属性 -DANDROID_STL 的 STL,并将其放入 c++_shared。有了这个属性,我的异常在某些架构上得到了正确处理。 但是在体系结构 armv7l 上(使用 System.getProperty("os.arch") 找到),throw 导致 SIGSEGV,这使我的应用程序崩溃。

这里是我的 build.gradle 文件的 android 部分:

android {
    compileSdkVersion Versions.COMPILE_SDK
    defaultConfig {
        applicationId "com.example.myapplication"
        minSdkVersion Versions.MIN_SDK
        targetSdkVersion Versions.TARGET_SDK
        versionCode 1
        versionName "1.0"
        externalNativeBuild {
            cmake {
                cppFlags "-v"
                arguments "-DANDROID_CPP_FEATURES=rtti exceptions", "-DANDROID_STL=c++_shared"
            }
        }
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
    externalNativeBuild {
        cmake {
            path "CMakeLists.txt"
        }
    }
    sourceSets {
        main.jniLibs.srcDirs += 'src/main/jniLibs'
    }
}

这是我的 CMakeLists.txt 文件:

# For more information about using CMake with Android Studio, read the
# documentation: https://d.android.com/studio/projects/add-native-code.html

# Sets the minimum version of CMake required to build the native library.

cmake_minimum_required(VERSION 3.4.1)

# Library name
set(library_name native-lib)

# Path to djinni support code
set(support_dir src/djinni/support-lib/jni)
# Path to generated code and own c++ implementation
set(include_dirs src/djinni/generated/jni src/djinni/generated/cpp src/main/cpp)

# Djinni support code that needs to be compiled
file(
        GLOB_RECURSE support_srcs
        ${support_dir}/*.cpp)

# Generated code and c++ implementations that needs to be compiled
file(
        GLOB_RECURSE lib_srcs
        src/djinni/generated/cpp/*.cpp
        src/djinni/generated/jni/*.cpp
        src/main/cpp/*.cpp)


# All the implementation files that make up our library
set(complete_srcs ${support_srcs} ${lib_srcs})

# Define library referring to the sources above
add_library(${library_name} SHARED ${complete_srcs})


# Add OpenCV library
set(opencv_library_name opencv)
set(opencv_dir src/main/jniLibs/${CMAKE_ANDROID_ARCH_ABI})

add_library(${opencv_library_name} SHARED IMPORTED)
set_target_properties(${opencv_library_name} PROPERTIES IMPORTED_LOCATION ${PROJECT_SOURCE_DIR}/${opencv_dir}/libopencv_java3.so)

# We link opencv to our datamatrix library
target_link_libraries(${library_name} ${opencv_library_name})

# Define INCLUDE DIRECTORIES property for native-lib
target_include_directories(${library_name} PUBLIC ${include_dirs} ${support_dir})

这是在我的应用程序构建过程中构建单个 C++ 文件的命令(使用 cppFlags 中的 -v 找到):

 "/Users/me/Library/Android/sdk/ndk-bundle/toolchains/llvm/prebuilt/darwin-x86_64/bin/clang++" -cc1 -triple i686-none-linux-android21 -emit-obj -mrelax-all -mnoexecstack -disable-free -disable-llvm-verifier -discard-value-names -main-file-name usercodedecoder.cpp -mrelocation-model pic -pic-level 2 -mthread-model posix -mdisable-fp-elim -fmath-errno -masm-verbose -mconstructor-aliases -munwind-tables -fuse-init-array -target-cpu i686 -target-feature +ssse3 -dwarf-column-info -debug-info-kind=standalone -dwarf-version=4 -debugger-tuning=gdb -target-linker-version 241.9 -v -ffunction-sections -fdata-sections -coverage-notes-file /Users/me/Documents/workspace/my_project/app/.externalNativeBuild/cmake/debug/x86/CMakeFiles/native-lib.dir/src/main/cpp/usercodedecoder.cpp.gcno -resource-dir /Users/me/Library/Android/sdk/ndk-bundle/toolchains/llvm/prebuilt/darwin-x86_64/lib64/clang/8.0.2 -dependency-file CMakeFiles/native-lib.dir/src/main/cpp/usercodedecoder.cpp.o.d -sys-header-deps -MT CMakeFiles/native-lib.dir/src/main/cpp/usercodedecoder.cpp.o -D datamatrix_native_lib_EXPORTS -I ../../../../src/djinni/generated/jni -I ../../../../src/djinni/generated/cpp -I ../../../../src/main/cpp -I ../../../../src/djinni/support-lib/jni -D ANDROID -isysroot /Users/me/Library/Android/sdk/ndk-bundle/toolchains/llvm/prebuilt/darwin-x86_64/sysroot -internal-isystem /Users/me/Library/Android/sdk/ndk-bundle/toolchains/llvm/prebuilt/darwin-x86_64/sysroot/usr/include/c++/v1 -internal-isystem /Users/me/Library/Android/sdk/ndk-bundle/toolchains/llvm/prebuilt/darwin-x86_64/sysroot/usr/local/include -internal-isystem /Users/me/Library/Android/sdk/ndk-bundle/toolchains/llvm/prebuilt/darwin-x86_64/lib64/clang/8.0.2/include -internal-externc-isystem /Users/me/Library/Android/sdk/ndk-bundle/toolchains/llvm/prebuilt/darwin-x86_64/sysroot/usr/include/i686-linux-android -internal-externc-isystem /Users/me/Library/Android/sdk/ndk-bundle/toolchains/llvm/prebuilt/darwin-x86_64/sysroot/include -internal-externc-isystem /Users/me/Library/Android/sdk/ndk-bundle/toolchains/llvm/prebuilt/darwin-x86_64/sysroot/usr/include -O0 -Wformat -Werror=format-security -fdeprecated-macro -fdebug-compilation-dir /Users/me/Documents/workspace/my_project/app/.externalNativeBuild/cmake/debug/x86 -ferror-limit 19 -fmessage-length 0 -stack-protector 2 -mstackrealign -fobjc-runtime=gcc -fcxx-exceptions -fexceptions -fdiagnostics-show-option -o CMakeFiles/native-lib.dir/src/main/cpp/usercodedecoder.cpp.o -x c++ ../../../../src/main/cpp/usercodedecoder.cpp

我对这类问题做了一些研究,但找不到可以帮助我消除某些架构上的 SIGSEGV 错误的东西。

我想问题出在我的构建过程中,但我不知道该怎么做才能解决它。

最佳答案

opencv 可能构建不正确。参见 https://github.com/android-ndk/ndk/issues/889有关诊断的说明。只需对您的 opencv 库执行与该错误建议的 qt 相同的操作即可。

关于Android C++ 异常抛出导致 SIGSEGV,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/54351887/

相关文章:

android - 在listview android中对多行进行分组

android - 如何实时获取拨出电话时长?

java - 更改android应用程序自适应可绘制对象?

c++ - 将帧速率提高到 30 fps 并从 AVT GigE 相机获取 bayerrg8

c++ - 如何定义一个函数指针,它采用与给定函数相同的参数和返回值?

linux - Cpack header 文件夹目标

android - 在 Android Studio 3.0 中构建 PIE 和非 PIE 可执行文件

android - 警告:需要版本2.10。当前版本是2.4

c++ - 从相机中提取的异步数据会产生随机崩溃

cmake - CMake链接到外部库